robosats/api/admin.py

483 lines
15 KiB
Python
Raw Permalink Normal View History

2022-10-25 18:04:12 +00:00
from statistics import median
from django.contrib import admin, messages
2022-01-04 15:58:10 +00:00
from django.contrib.auth.admin import UserAdmin
2022-10-25 18:04:12 +00:00
from django.contrib.auth.models import Group, User
from django.utils.html import format_html
2022-10-25 18:04:12 +00:00
from django_admin_relation_links import AdminChangeLinksMixin
from rest_framework.authtoken.admin import TokenAdmin
from rest_framework.authtoken.models import TokenProxy
2022-10-25 18:04:12 +00:00
from api.logics import Logics
from api.models import Currency, LNPayment, MarketTick, OnchainPayment, Order, Robot
from api.utils import objects_to_hyperlinks
from api.tasks import send_notification
2022-10-20 09:56:10 +00:00
2022-01-04 15:58:10 +00:00
admin.site.unregister(Group)
admin.site.unregister(User)
admin.site.unregister(TokenProxy)
2022-01-04 15:58:10 +00:00
2022-02-17 19:50:10 +00:00
class RobotInline(admin.StackedInline):
model = Robot
2022-02-17 19:50:10 +00:00
can_delete = False
2022-10-20 09:56:10 +00:00
fields = ("avatar_tag",)
2022-02-17 19:50:10 +00:00
readonly_fields = ["avatar_tag"]
2022-03-18 21:21:13 +00:00
show_change_link = True
2022-02-17 19:50:10 +00:00
2022-10-20 09:56:10 +00:00
2022-01-04 15:58:10 +00:00
# extended users with avatars
@admin.register(User)
2022-03-18 21:21:13 +00:00
class EUserAdmin(AdminChangeLinksMixin, UserAdmin):
inlines = [RobotInline]
2022-02-17 19:50:10 +00:00
list_display = (
"avatar_tag",
"id",
"robot_link",
2022-02-17 19:50:10 +00:00
"username",
"last_login",
"date_joined",
"is_staff",
)
list_display_links = ("id", "username")
change_links = ("robot",)
2022-10-20 09:56:10 +00:00
ordering = ("-id",)
2022-02-17 19:50:10 +00:00
2022-01-04 15:58:10 +00:00
def avatar_tag(self, obj):
return obj.robot.avatar_tag()
2022-01-04 15:58:10 +00:00
2022-10-20 09:56:10 +00:00
# extended tokens with raw id fields and avatars
@admin.register(TokenProxy)
class ETokenAdmin(AdminChangeLinksMixin, TokenAdmin):
raw_id_fields = ["user"]
list_display = (
"avatar_tag",
"key",
"user_link",
)
list_display_links = ("key",)
change_links = ("user",)
def avatar_tag(self, obj):
return obj.user.robot.avatar_tag()
@admin.register(Order)
2022-01-05 11:20:08 +00:00
class OrderAdmin(AdminChangeLinksMixin, admin.ModelAdmin):
2022-02-17 19:50:10 +00:00
list_display = (
"id",
"type",
"maker_link",
"taker_link",
"status",
"amt",
2022-02-17 19:50:10 +00:00
"currency_link",
"t0_satoshis",
"is_disputed",
"is_fiat_sent",
"created_at",
"expires_at",
2022-06-07 22:14:56 +00:00
"payout_tx_link",
2022-02-17 19:50:10 +00:00
"payout_link",
"maker_bond_link",
"taker_bond_link",
"trade_escrow_link",
)
list_display_links = ("id", "type")
change_links = (
"maker",
"taker",
"currency",
2022-06-07 22:14:56 +00:00
"payout_tx",
2022-02-17 19:50:10 +00:00
"payout",
"maker_bond",
"taker_bond",
"trade_escrow",
)
raw_id_fields = (
"maker",
"taker",
"payout_tx",
"payout",
"maker_bond",
"taker_bond",
"trade_escrow",
)
2022-10-20 09:56:10 +00:00
list_filter = (
"is_disputed",
"is_fiat_sent",
"is_swap",
"type",
"currency",
"status",
)
search_fields = [
"id",
"reference",
2023-05-02 11:04:24 +00:00
"maker__username",
"taker__username",
"amount",
2023-05-02 11:04:24 +00:00
"payout__payment_hash",
"maker_bond__payment_hash",
"taker_bond__payment_hash",
"trade_escrow__payment_hash",
"payout_tx__txid",
"payout_tx__address",
"min_amount",
"max_amount",
]
readonly_fields = ("reference", "_logs")
def _logs(self, obj):
if not obj.logs:
return format_html("<b>No logs were recorded</b>")
with_hyperlinks = objects_to_hyperlinks(obj.logs)
2024-01-30 16:22:57 +00:00
try:
html_logs = format_html(
f'<table style="width: 100%">{with_hyperlinks}</table>'
)
except Exception as e:
html_logs = f"An error occurred while formatting the parsed logs as HTML. Exception {e}"
return html_logs
2022-10-20 09:56:10 +00:00
actions = [
"cancel_public_order",
2022-10-20 09:56:10 +00:00
"maker_wins",
"taker_wins",
"return_everything",
"successful_trade",
2023-05-02 11:04:24 +00:00
"compute_median_trade_time",
2022-10-20 09:56:10 +00:00
]
@admin.action(description="Close public order")
def cancel_public_order(self, request, queryset):
"""
Closes an existing Public/Paused order.
"""
for order in queryset:
if order.status in [Order.Status.PUB, Order.Status.PAU]:
if Logics.return_bond(order.maker_bond):
order.update_status(Order.Status.UCA)
self.message_user(
request,
f"Order {order.id} successfully closed",
messages.SUCCESS,
)
send_notification.delay(
order_id=order.id, message="coordinator_cancelled"
)
else:
self.message_user(
request,
f"Could not unlock bond of {order.id}",
messages.ERROR,
)
else:
self.message_user(
request,
f"Order {order.id} is not public or paused",
messages.ERROR,
)
2022-10-20 09:56:10 +00:00
@admin.action(description="Solve dispute: maker wins")
def maker_wins(self, request, queryset):
2022-10-20 09:56:10 +00:00
"""
Solves a dispute on favor of the maker.
Adds Sats to compensations (earned_rewards) of the maker robot.
2022-10-20 09:56:10 +00:00
"""
for order in queryset:
2022-10-20 09:56:10 +00:00
if (
order.status in [Order.Status.DIS, Order.Status.WFR]
and order.is_disputed
):
own_bond_sats = order.maker_bond.num_satoshis
if Logics.is_buyer(order, order.maker):
if order.is_swap:
trade_sats = order.payout_tx.num_satoshis
else:
trade_sats = order.payout.num_satoshis
else:
trade_sats = order.trade_escrow.num_satoshis
order.maker.robot.earned_rewards = own_bond_sats + trade_sats
order.maker.robot.save(update_fields=["earned_rewards"])
order.update_status(Order.Status.TLD)
2022-10-20 09:56:10 +00:00
self.message_user(
request,
f"Dispute of order {order.id} solved successfully on favor of the maker",
messages.SUCCESS,
)
else:
2022-10-20 09:56:10 +00:00
self.message_user(
request,
f"Order {order.id} is not in a disputed state",
messages.ERROR,
)
2022-10-20 09:56:10 +00:00
@admin.action(description="Solve dispute: taker wins")
def taker_wins(self, request, queryset):
2022-10-20 09:56:10 +00:00
"""
Solves a dispute on favor of the taker.
Adds Sats to compensations (earned_rewards) of the taker robot.
2022-10-20 09:56:10 +00:00
"""
for order in queryset:
2022-10-20 09:56:10 +00:00
if (
order.status in [Order.Status.DIS, Order.Status.WFR]
and order.is_disputed
):
own_bond_sats = order.maker_bond.num_satoshis
if Logics.is_buyer(order, order.taker):
if order.is_swap:
trade_sats = order.payout_tx.num_satoshis
else:
trade_sats = order.payout.num_satoshis
else:
trade_sats = order.trade_escrow.num_satoshis
order.taker.robot.earned_rewards = own_bond_sats + trade_sats
order.taker.robot.save(update_fields=["earned_rewards"])
order.update_status(Order.Status.MLD)
2022-10-20 09:56:10 +00:00
self.message_user(
request,
f"Dispute of order {order.id} solved successfully on favor of the taker",
messages.SUCCESS,
)
else:
2022-10-20 09:56:10 +00:00
self.message_user(
request,
f"Order {order.id} is not in a disputed state",
messages.ERROR,
)
2022-10-20 09:56:10 +00:00
@admin.action(description="Solve dispute: return everything")
def return_everything(self, request, queryset):
2022-10-20 09:56:10 +00:00
"""
Solves a dispute by pushing back every bond and escrow to their sender.
2022-10-20 09:56:10 +00:00
"""
for order in queryset:
2022-10-20 09:56:10 +00:00
if (
order.status in [Order.Status.DIS, Order.Status.WFR]
and order.is_disputed
):
order.maker_bond.sender.robot.earned_rewards += (
2022-10-20 09:56:10 +00:00
order.maker_bond.num_satoshis
)
order.maker_bond.sender.robot.save(update_fields=["earned_rewards"])
order.taker_bond.sender.robot.earned_rewards += (
2022-10-20 09:56:10 +00:00
order.taker_bond.num_satoshis
)
order.taker_bond.sender.robot.save(update_fields=["earned_rewards"])
order.trade_escrow.sender.robot.earned_rewards += (
2022-10-20 09:56:10 +00:00
order.trade_escrow.num_satoshis
)
order.trade_escrow.sender.robot.save(update_fields=["earned_rewards"])
order.update_status(Order.Status.CCA)
2022-10-20 09:56:10 +00:00
self.message_user(
request,
f"Dispute of order {order.id} solved successfully, everything returned as compensations",
messages.SUCCESS,
)
else:
2022-10-20 09:56:10 +00:00
self.message_user(
request,
f"Order {order.id} is not in a disputed state",
messages.ERROR,
)
@admin.action(description="Solve dispute: successful trade")
def successful_trade(self, request, queryset):
"""
Solves a dispute as if the trade had been successful, i.e.,
returns both bonds (added as compensations) and triggers the payout.
"""
for order in queryset:
if (
order.status in [Order.Status.DIS, Order.Status.WFR]
and order.is_disputed
):
order.maker.robot.earned_rewards = order.maker_bond.num_satoshis
order.maker.robot.save(update_fields=["earned_rewards"])
order.taker.robot.earned_rewards = order.taker_bond.num_satoshis
order.taker.robot.save(update_fields=["earned_rewards"])
if order.is_swap:
order.payout_tx.status = OnchainPayment.Status.VALID
order.payout_tx.save(update_fields=["status"])
order.update_status(Order.Status.SUC)
else:
order.update_status(Order.Status.PAY)
Logics.pay_buyer(order)
self.message_user(
request,
f"Dispute of order {order.id} solved as successful trade",
messages.SUCCESS,
)
else:
self.message_user(
request,
f"Order {order.id} is not in a disputed state",
messages.ERROR,
)
2022-10-20 09:56:10 +00:00
@admin.action(description="Compute median trade completion time")
2023-05-02 11:04:24 +00:00
def compute_median_trade_time(self, request, queryset):
2022-10-20 09:56:10 +00:00
"""
Computes the median time from an order taken to finishing
successfully for the set of selected orders.
2022-10-20 09:56:10 +00:00
"""
times = []
for order in queryset:
if order.contract_finalization_time:
timedelta = order.contract_finalization_time - order.last_satoshis_time
times.append(timedelta.total_seconds())
2022-10-20 09:56:10 +00:00
if len(times) > 0:
median_time_secs = median(times)
2022-10-20 09:56:10 +00:00
mins = int(median_time_secs / 60)
secs = int(median_time_secs - mins * 60)
self.message_user(
request,
f"The median time to complete the trades is {mins}m {secs}s",
messages.SUCCESS,
)
else:
2022-10-20 09:56:10 +00:00
self.message_user(
request,
"There is no successfully finished orders in the selection",
messages.ERROR,
)
def amt(self, obj):
if obj.has_range and obj.amount is None:
2022-10-20 09:56:10 +00:00
return str(float(obj.min_amount)) + "-" + str(float(obj.max_amount))
else:
2022-10-20 09:56:10 +00:00
return float(obj.amount)
@admin.register(LNPayment)
2022-01-05 11:20:08 +00:00
class LNPaymentAdmin(AdminChangeLinksMixin, admin.ModelAdmin):
2022-02-17 19:50:10 +00:00
list_display = (
"hash",
"concept",
"status",
"num_satoshis",
"fee",
2022-02-17 19:50:10 +00:00
"type",
"expires_at",
"expiry_height",
"sender_link",
"receiver_link",
"order_made_link",
"order_taken_link",
"order_escrow_link",
2022-06-19 06:09:21 +00:00
"order_paid_LN_link",
"order_donated_link",
2022-02-17 19:50:10 +00:00
)
list_display_links = ("hash", "concept")
change_links = (
"sender",
"receiver",
"order_made",
"order_taken",
"order_escrow",
2022-06-19 06:09:21 +00:00
"order_paid_LN",
"order_donated",
2022-02-17 19:50:10 +00:00
)
2023-04-27 09:38:16 +00:00
raw_id_fields = (
"receiver",
"sender",
"order_donated",
2023-04-27 09:38:16 +00:00
)
2022-02-17 19:50:10 +00:00
list_filter = ("type", "concept", "status")
2022-10-20 09:56:10 +00:00
ordering = ("-expires_at",)
search_fields = [
"payment_hash",
2023-05-02 11:04:24 +00:00
"preimage",
2022-10-20 09:56:10 +00:00
"num_satoshis",
"sender__username",
"receiver__username",
"description",
]
2022-02-17 19:50:10 +00:00
2022-06-07 22:14:56 +00:00
@admin.register(OnchainPayment)
class OnchainPaymentAdmin(AdminChangeLinksMixin, admin.ModelAdmin):
list_display = (
"id",
"address",
"concept",
"status",
"broadcasted",
2022-06-07 22:14:56 +00:00
"num_satoshis",
"hash",
"swap_fee_rate",
"mining_fee_sats",
"balance_link",
"order_paid_TX_link",
2022-06-07 22:14:56 +00:00
)
change_links = (
"balance",
"order_paid_TX",
2022-06-07 22:14:56 +00:00
)
2023-04-27 09:38:16 +00:00
raw_id_fields = (
"receiver",
"balance",
)
2022-10-20 09:56:10 +00:00
list_display_links = ("id", "address", "concept")
2022-06-07 22:14:56 +00:00
list_filter = ("concept", "status")
2022-10-20 09:56:10 +00:00
search_fields = ["address", "num_satoshis", "receiver__username", "txid"]
@admin.register(Robot)
class UserRobotAdmin(AdminChangeLinksMixin, admin.ModelAdmin):
2022-02-17 19:50:10 +00:00
list_display = (
"avatar_tag",
"id",
"user_link",
"telegram_enabled",
2022-02-17 19:50:10 +00:00
"total_contracts",
"earned_rewards",
"claimed_rewards",
2022-02-17 19:50:10 +00:00
"platform_rating",
"num_disputes",
"lost_disputes",
)
raw_id_fields = ("user",)
list_editable = ["earned_rewards"]
2022-02-17 19:50:10 +00:00
list_display_links = ("avatar_tag", "id")
change_links = ["user"]
readonly_fields = ["avatar_tag"]
2022-10-20 09:56:10 +00:00
search_fields = ["user__username", "id"]
readonly_fields = ("hash_id", "public_key", "encrypted_private_key")
2022-02-17 19:50:10 +00:00
@admin.register(Currency)
class CurrencieAdmin(admin.ModelAdmin):
2022-02-17 19:50:10 +00:00
list_display = ("id", "currency", "exchange_rate", "timestamp")
list_display_links = ("id", "currency")
readonly_fields = ("currency", "exchange_rate", "timestamp")
2022-10-20 09:56:10 +00:00
ordering = ("id",)
2022-02-17 19:50:10 +00:00
@admin.register(MarketTick)
class MarketTickAdmin(admin.ModelAdmin):
2022-10-20 09:56:10 +00:00
list_display = ("timestamp", "price", "volume", "premium", "currency", "fee")
readonly_fields = ("timestamp", "price", "volume", "premium", "currency", "fee")
2022-02-17 19:50:10 +00:00
list_filter = ["currency"]
2022-10-20 09:56:10 +00:00
ordering = ("-timestamp",)