Improve tasks reliability (#390)

* Add broadcasted flag

* Improve celery tasks reliability
This commit is contained in:
Reckless_Satoshi 2023-03-16 00:53:37 +00:00 committed by GitHub
parent 64f3243c53
commit 94bc44ad0e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 139 additions and 91 deletions

View File

@ -290,6 +290,7 @@ class OnchainPaymentAdmin(AdminChangeLinksMixin, admin.ModelAdmin):
"address", "address",
"concept", "concept",
"status", "status",
"broadcasted",
"num_satoshis", "num_satoshis",
"hash", "hash",
"swap_fee_rate", "swap_fee_rate",

View File

@ -158,7 +158,9 @@ class LNNode:
request, metadata=[("macaroon", MACAROON.hex())] request, metadata=[("macaroon", MACAROON.hex())]
) )
if response.txid:
onchainpayment.txid = response.txid onchainpayment.txid = response.txid
onchainpayment.broadcasted = True
onchainpayment.save() onchainpayment.save()
return True return True

View File

@ -152,26 +152,42 @@ class Command(BaseCommand):
queryset_retries = LNPayment.objects.filter( queryset_retries = LNPayment.objects.filter(
type=LNPayment.Types.NORM, type=LNPayment.Types.NORM,
status__in=[LNPayment.Status.VALIDI, LNPayment.Status.FAILRO], status=LNPayment.Status.FAILRO,
in_flight=False, in_flight=False,
routing_attempts__in=[1, 2],
last_routing_time__lt=( last_routing_time__lt=(
timezone.now() - timedelta(minutes=int(config("RETRY_TIME"))) timezone.now() - timedelta(minutes=int(config("RETRY_TIME")))
), ),
) )
queryset = queryset.union(queryset_retries) # Payments that still have the in_flight flag whose last payment attempt was +3 min ago
# are probably stuck. We retry them. The follow_send_invoice() task can also do TrackPaymentV2 if the
# previous attempt is still ongoing
queryset_stuck = LNPayment.objects.filter(
type=LNPayment.Types.NORM,
status__in=[LNPayment.Status.FAILRO, LNPayment.Status.FLIGHT],
in_flight=True,
last_routing_time__lt=(timezone.now() - timedelta(minutes=3)),
)
queryset = queryset.union(queryset_retries).union(queryset_stuck)
if len(queryset) > 0:
for lnpayment in queryset: for lnpayment in queryset:
# Checks that this onchain payment is part of an order with a settled escrow
if not hasattr(lnpayment, "order_paid_LN"):
self.stdout.write(f"Ln payment {str(lnpayment)} has no parent order!")
return
order = lnpayment.order_paid_LN
if order.trade_escrow.status == LNPayment.Status.SETLED:
follow_send_payment.delay(lnpayment.payment_hash) follow_send_payment.delay(lnpayment.payment_hash)
def send_onchain_payments(self): def send_onchain_payments(self):
queryset = OnchainPayment.objects.filter( queryset = OnchainPayment.objects.filter(
status=OnchainPayment.Status.QUEUE, status=OnchainPayment.Status.QUEUE,
broadcasted=False,
) )
if len(queryset) > 0:
for onchainpayment in queryset: for onchainpayment in queryset:
# Checks that this onchain payment is part of an order with a settled escrow # Checks that this onchain payment is part of an order with a settled escrow
if not hasattr(onchainpayment, "order_paid_TX"): if not hasattr(onchainpayment, "order_paid_TX"):
@ -180,13 +196,18 @@ class Command(BaseCommand):
) )
return return
order = onchainpayment.order_paid_TX order = onchainpayment.order_paid_TX
if order.trade_escrow.status == LNPayment.Status.SETLED: if (
order.trade_escrow.status == LNPayment.Status.SETLED
and order.trade_escrow.num_satoshis >= onchainpayment.num_satoshis
):
# Sends out onchainpayment # Sends out onchainpayment
LNNode.pay_onchain( LNNode.pay_onchain(
onchainpayment, onchainpayment,
OnchainPayment.Status.QUEUE, OnchainPayment.Status.QUEUE,
OnchainPayment.Status.MEMPO, OnchainPayment.Status.MEMPO,
) )
onchainpayment.save()
else: else:
self.stdout.write( self.stdout.write(
f"Onchain payment {str(onchainpayment)} for order {str(order)} escrow is not settled!" f"Onchain payment {str(onchainpayment)} for order {str(order)} escrow is not settled!"

View File

@ -203,6 +203,8 @@ class OnchainPayment(models.Model):
choices=Status.choices, null=False, default=Status.CREAT choices=Status.choices, null=False, default=Status.CREAT
) )
broadcasted = models.BooleanField(default=False, null=False, blank=False)
# payment info # payment info
address = models.CharField( address = models.CharField(
max_length=100, unique=False, default=None, null=True, blank=True max_length=100, unique=False, default=None, null=True, blank=True

View File

@ -86,6 +86,8 @@ def follow_send_payment(hash):
from api.models import LNPayment, Order from api.models import LNPayment, Order
lnpayment = LNPayment.objects.get(payment_hash=hash) lnpayment = LNPayment.objects.get(payment_hash=hash)
lnpayment.last_routing_time = timezone.now()
lnpayment.save()
# Default is 0ppm. Set by the user over API. Client's default is 1000 ppm. # Default is 0ppm. Set by the user over API. Client's default is 1000 ppm.
fee_limit_sat = int( fee_limit_sat = int(
@ -101,29 +103,27 @@ def follow_send_payment(hash):
) )
order = lnpayment.order_paid_LN order = lnpayment.order_paid_LN
try: if order.trade_escrow.num_satoshis < lnpayment.num_satoshis:
for response in LNNode.routerstub.SendPaymentV2( print(f"Order: {order.id} Payout is larger than collateral !?")
request, metadata=[("macaroon", MACAROON.hex())] return
):
lnpayment.in_flight = True def handle_response(response):
lnpayment.save()
if response.status == 0: # Status 0 'UNKNOWN'
# Not sure when this status happens
lnpayment.in_flight = False
lnpayment.save()
if response.status == 1: # Status 1 'IN_FLIGHT'
print("IN_FLIGHT")
lnpayment.status = LNPayment.Status.FLIGHT lnpayment.status = LNPayment.Status.FLIGHT
lnpayment.in_flight = True lnpayment.in_flight = True
lnpayment.save() lnpayment.save()
order.status = Order.Status.PAY order.status = Order.Status.PAY
order.save() order.save()
if response.status == 0: # Status 0 'UNKNOWN'
# Not sure when this status happens
print(f"Order: {order.id} UNKNOWN. Hash {hash}")
lnpayment.in_flight = False
lnpayment.save()
if response.status == 1: # Status 1 'IN_FLIGHT'
print(f"Order: {order.id} IN_FLIGHT. Hash {hash}")
if response.status == 3: # Status 3 'FAILED' if response.status == 3: # Status 3 'FAILED'
print("FAILED")
lnpayment.status = LNPayment.Status.FAILRO lnpayment.status = LNPayment.Status.FAILRO
lnpayment.last_routing_time = timezone.now() lnpayment.last_routing_time = timezone.now()
lnpayment.routing_attempts += 1 lnpayment.routing_attempts += 1
@ -139,22 +139,16 @@ def follow_send_payment(hash):
seconds=order.t_to_expire(Order.Status.FAI) seconds=order.t_to_expire(Order.Status.FAI)
) )
order.save() order.save()
context = { print(
"routing_failed": LNNode.payment_failure_context[ f"Order: {order.id} FAILED. Hash: {hash} Reason: {LNNode.payment_failure_context[response.failure_reason]}"
response.failure_reason )
], return {
"IN_FLIGHT": False, "succeded": False,
"context": f"payment failure reason: {LNNode.payment_failure_context[response.failure_reason]}",
} }
# If failed due to not route, reset mission control. (This won't scale well, just a temporary fix)
# ResetMC deactivate temporary for tests
# if response.failure_reason==2:
# LNNode.resetmc()
return False, context
if response.status == 2: # Status 2 'SUCCEEDED' if response.status == 2: # Status 2 'SUCCEEDED'
print("SUCCEEDED") print(f"SUCCEEDED. Order: {order.id}. Hash: {hash}")
lnpayment.status = LNPayment.Status.SUCCED lnpayment.status = LNPayment.Status.SUCCED
lnpayment.fee = float(response.fee_msat) / 1000 lnpayment.fee = float(response.fee_msat) / 1000
lnpayment.preimage = response.payment_preimage lnpayment.preimage = response.payment_preimage
@ -164,11 +158,20 @@ def follow_send_payment(hash):
seconds=order.t_to_expire(Order.Status.SUC) seconds=order.t_to_expire(Order.Status.SUC)
) )
order.save() order.save()
return True, None results = {"succeded": True}
return results
try:
for response in LNNode.routerstub.SendPaymentV2(
request, metadata=[("macaroon", MACAROON.hex())]
):
handle_response(response)
except Exception as e: except Exception as e:
if "invoice expired" in str(e): if "invoice expired" in str(e):
print("INVOICE EXPIRED") print(f"Order: {order.id}. INVOICE EXPIRED. Hash: {hash}")
lnpayment.status = LNPayment.Status.EXPIRE lnpayment.status = LNPayment.Status.EXPIRE
lnpayment.last_routing_time = timezone.now() lnpayment.last_routing_time = timezone.now()
lnpayment.in_flight = False lnpayment.in_flight = False
@ -178,8 +181,20 @@ def follow_send_payment(hash):
seconds=order.t_to_expire(Order.Status.FAI) seconds=order.t_to_expire(Order.Status.FAI)
) )
order.save() order.save()
context = {"routing_failed": "The payout invoice has expired"} results = {"succeded": False, "context": "The payout invoice has expired"}
return False, context return results
if "payment is in transition" in str(e):
print(f"Order: {order.id}. ALREADY IN TRANSITION. Hash: {hash}.")
request = LNNode.routerrpc.TrackPaymentRequest(
payment_hash=bytes.fromhex(hash)
)
for response in LNNode.routerstub.TrackPaymentV2(
request, metadata=[("macaroon", MACAROON.hex())]
):
handle_response(response)
@shared_task(name="payments_cleansing") @shared_task(name="payments_cleansing")

View File

@ -100,7 +100,7 @@ services:
volumes: volumes:
- .:/usr/src/robosats - .:/usr/src/robosats
- ./node/lnd:/lnd - ./node/lnd:/lnd
command: celery -A robosats worker --loglevel=WARNING command: celery -A robosats worker --loglevel=INFO --concurrency 4 --max-tasks-per-child=4 --max-memory-per-child=200000
depends_on: depends_on:
- redis - redis
network_mode: service:tor network_mode: service:tor

View File

@ -1,4 +1,4 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useLayoutEffect, useState } from 'react';
import { useTranslation } from 'react-i18next'; import { useTranslation } from 'react-i18next';
import { Button, Tooltip, TextField, Grid, Paper } from '@mui/material'; import { Button, Tooltip, TextField, Grid, Paper } from '@mui/material';
import { encryptMessage, decryptMessage } from '../../../../pgp'; import { encryptMessage, decryptMessage } from '../../../../pgp';
@ -73,6 +73,13 @@ const EncryptedSocketChat: React.FC<Props> = ({
} }
}, [status]); }, [status]);
useLayoutEffect(() => {
// On component unmount close reconnecting-websockets
return () => {
connection?.close();
};
}, []);
useEffect(() => { useEffect(() => {
if (messages.length > messageCount) { if (messages.length > messageCount) {
audio.play(); audio.play();

View File

@ -9,7 +9,7 @@ django-timezone-field==4.2.3
djangorestframework==3.13.1 djangorestframework==3.13.1
channels==3.0.4 channels==3.0.4
channels-redis==3.3.1 channels-redis==3.3.1
celery==5.2.3 celery==5.2.7
grpcio==1.43.0 grpcio==1.43.0
googleapis-common-protos==1.53.0 googleapis-common-protos==1.53.0
grpcio-tools==1.43.0 grpcio-tools==1.43.0
@ -21,7 +21,7 @@ ring==0.9.1
git+https://github.com/Reckless-Satoshi/Robohash.git git+https://github.com/Reckless-Satoshi/Robohash.git
scipy==1.8.0 scipy==1.8.0
gunicorn==20.1.0 gunicorn==20.1.0
psycopg2==2.9.3 psycopg2==2.9.5
SQLAlchemy==1.4.31 SQLAlchemy==1.4.31
django-import-export==2.7.1 django-import-export==2.7.1
requests[socks] requests[socks]