from django.template.defaultfilters import default from rest_framework import serializers from .models import MarketTick, Order from decouple import config RETRY_TIME = int(config("RETRY_TIME")) MIN_PUBLIC_ORDER_DURATION_SECS=60*60*float(config("MIN_PUBLIC_ORDER_DURATION")) MAX_PUBLIC_ORDER_DURATION_SECS=60*60*float(config("MAX_PUBLIC_ORDER_DURATION")) class InfoSerializer(serializers.Serializer): num_public_buy_orders = serializers.IntegerField() num_public_sell_orders = serializers.IntegerField() book_liquidity = serializers.IntegerField(help_text='Total amount of BTC in the order book') active_robots_today = serializers.CharField() last_day_nonkyc_btc_premium = serializers.FloatField(help_text='Average premium (weighted by volume) of the orders in the last 24h') last_day_volume = serializers.FloatField(help_text='Total volume in BTC in the last 24h') lifetime_volume = serializers.FloatField(help_text='Total volume in BTC since exchange\'s inception') lnd_version = serializers.CharField() robosats_running_commit_hash = serializers.CharField() alternative_site = serializers.CharField() alternative_name = serializers.CharField() node_alias = serializers.CharField() node_id = serializers.CharField() network = serializers.CharField() maker_fee = serializers.FloatField(help_text='Exchange\'s set maker fee') taker_fee = serializers.FloatField(help_text='Exchange\'s set taker fee ') bond_size = serializers.FloatField(help_text='Default bond size (percent)') current_swap_fee_rate = serializers.FloatField(help_text='Swap fees to perform on-chain transaction (percent)') nickname = serializers.CharField(help_text='Currenlty logged in Robot name') referral_code = serializers.CharField(help_text='Logged in users\'s referral code') earned_rewards = serializers.IntegerField(help_text='Logged in user\'s earned rewards in satoshis') class ListOrderSerializer(serializers.ModelSerializer): class Meta: model = Order fields = ( "id", "status", "created_at", "expires_at", "type", "currency", "amount", "has_range", "min_amount", "max_amount", "payment_method", "is_explicit", "premium", "satoshis", "bondless_taker", "maker", "taker", "escrow_duration", "bond_size" ) # Only used in oas_schemas class SummarySerializer(serializers.Serializer): sent_fiat = serializers.IntegerField( required=False, help_text="same as `amount` (only for buyer)" ) received_sats = serializers.IntegerField( required=False, help_text="same as `trade_satoshis` (only for buyer)" ) is_swap = serializers.BooleanField( required=False, help_text="True if the payout was on-chain (only for buyer)" ) received_onchain_sats = serializers.IntegerField( required=False, help_text="The on-chain sats received (only for buyer and if `is_swap` is `true`)" ) mining_fee_sats = serializers.IntegerField( required=False, help_text="Mining fees paid in satoshis (only for buyer and if `is_swap` is `true`)" ) swap_fee_sats = serializers.IntegerField( required=False, help_text="Exchange swap fee in sats (i.e excluding miner fees) (only for buyer and if `is_swap` is `true`)" ) swap_fee_percent = serializers.FloatField( required=False, help_text="same as `swap_fee_rate` (only for buyer and if `is_swap` is `true`" ) sent_sats = serializers.IntegerField( required=False, help_text="The total sats you sent (only for seller)" ) received_fiat = serializers.IntegerField( required=False, help_text="same as `amount` (only for seller)" ) trade_fee_sats = serializers.IntegerField( required=False, help_text="Exchange fees in sats (Does not include swap fee and miner fee)" ) # Only used in oas_schemas class PlatformSummarySerializer(serializers.Serializer): contract_timestamp = serializers.DateTimeField( required=False, help_text="Timestamp of when the contract was finalized (price and sats fixed)" ) contract_total_time = serializers.FloatField( required=False, help_text="The time taken for the contract to complete (from taker taking the order to completion of order) in seconds" ) routing_fee_sats = serializers.IntegerField( required=False, help_text="Sats payed by the exchange for routing fees. Mining fee in case of on-chain swap payout" ) trade_revenue_sats = serializers.IntegerField( required=False, help_text="The sats the exchange earned from the trade" ) # Only used in oas_schemas class OrderDetailSerializer(serializers.ModelSerializer): total_secs_exp = serializers.IntegerField( required=False, help_text="Duration of time (in seconds) to expire, according to the current status of order." "This is duration of time after `created_at` (in seconds) that the order will automatically expire." "This value changes according to which stage the order is in" ) penalty = serializers.DateTimeField( required=False, help_text="Time when the user penalty will expire. Penalty applies when you create orders repeatedly without commiting a bond" ) is_maker = serializers.BooleanField( required=False, help_text="Whether you are the maker or not" ) is_taker = serializers.BooleanField( required=False, help_text="Whether you are the taker or not" ) is_participant = serializers.BooleanField( required=False, help_text="True if you are either a taker or maker, False otherwise" ) maker_status = serializers.CharField( required=False, help_text="Status of the maker:\n" "- **'Active'** (seen within last 2 min)\n" "- **'Seen Recently'** (seen within last 10 min)\n" "- **'Inactive'** (seen more than 10 min ago)\n\n" "Note: When you make a request to this route, your own status get's updated and can be seen by your counterparty" ) taker_status = serializers.BooleanField( required=False, help_text="True if you are either a taker or maker, False otherwise" ) price_now = serializers.IntegerField( required=False, help_text="Price of the order in the order's currency at the time of request (upto 5 significant digits)" ) premium = serializers.IntegerField( required=False, help_text="Premium over the CEX price at the current time" ) premium_percentile = serializers.IntegerField( required=False, help_text="(Only if `is_maker`) Premium percentile of your order compared to other public orders in the same currency currently in the order book" ) num_similar_orders = serializers.IntegerField( required=False, help_text="(Only if `is_maker`) The number of public orders of the same currency currently in the order book" ) tg_enabled = serializers.BooleanField( required=False, help_text="(Only if `is_maker`) Whether Telegram notification is enabled or not" ) tg_token = serializers.CharField( required=False, help_text="(Only if `is_maker`) Your telegram bot token required to enable notifications." ) tg_bot_name = serializers.CharField( required=False, help_text="(Only if `is_maker`) The Telegram username of the bot" ) is_buyer = serializers.BooleanField( required=False, help_text="Whether you are a buyer of sats (you will be receiving sats)" ) is_seller = serializers.BooleanField( required=False, help_text="Whether you are a seller of sats or not (you will be sending sats)" ) maker_nick = serializers.CharField( required=False, help_text="Nickname (Robot name) of the maker" ) taker_nick = serializers.CharField( required=False, help_text="Nickname (Robot name) of the taker" ) status_message = serializers.CharField( required=False, help_text="The current status of the order corresponding to the `status`" ) is_fiat_sent = serializers.BooleanField( required=False, help_text="Whether or not the fiat amount is sent by the buyer" ) is_disputed = serializers.BooleanField( required=False, help_text="Whether or not the counterparty raised a dispute" ) ur_nick = serializers.CharField( required=False, help_text="Your Nickname" ) ur_nick = serializers.CharField( required=False, help_text="Your Nick" ) maker_locked = serializers.BooleanField( required=False, help_text="True if maker bond is locked, False otherwise" ) taker_locked = serializers.BooleanField( required=False, help_text="True if taker bond is locked, False otherwise" ) escrow_locked = serializers.BooleanField( required=False, help_text="True if escrow is locked, False otherwise. Escrow is the sats to be sold, held by Robosats until the trade is finised." ) trade_satoshis = serializers.IntegerField( required=False, help_text="Seller sees the amount of sats they need to send. Buyer sees the amount of sats they will receive " ) bond_invoice = serializers.CharField( required=False, help_text="When `status` = `0`, `3`. Bond invoice to be paid" ) bond_satoshis = serializers.IntegerField( required=False, help_text="The bond amount in satoshis" ) escrow_invoice = serializers.CharField( required=False, help_text="For the seller, the escrow invoice to be held by RoboSats" ) escrow_satoshis = serializers.IntegerField( required=False, help_text="The escrow amount in satoshis" ) invoice_amount = serializers.IntegerField( required=False, help_text="The amount in sats the buyer needs to submit an invoice of to receive the trade amount" ) swap_allowed = serializers.BooleanField( required=False, help_text="Whether on-chain swap is allowed" ) swap_failure_reason = serializers.CharField( required=False, help_text="Reason for why on-chain swap is not available" ) suggested_mining_fee_rate = serializers.IntegerField( required=False, help_text="fee in sats/vbyte for the on-chain swap" ) swap_fee_rate = serializers.FloatField( required=False, help_text="in percentage, the swap fee rate the platform charges" ) pending_cancel = serializers.BooleanField( required=False, help_text="Your counterparty requested for a collaborative cancel when `status` is either `8`, `9` or `10`" ) asked_for_cancel = serializers.BooleanField( required=False, help_text="You requested for a collaborative cancel `status` is either `8`, `9` or `10`" ) statement_submitted = serializers.BooleanField( required=False, help_text="True if you have submitted a statement. Available when `status` is `11`" ) retries = serializers.IntegerField( required=False, help_text="Number of times ln node has tried to make the payment to you (only if you are the buyer)" ) next_retry_time = serializers.DateTimeField( required=False, help_text=f"The next time payment will be retried. Payment is retried every {RETRY_TIME} sec" ) failure_reason = serializers.CharField( required=False, help_text="The reason the payout failed" ) invoice_expired = serializers.BooleanField( required=False, help_text="True if the payout invoice expired. `invoice_amount` will be re-set and sent which means the user has to submit a new invoice to be payed" ) trade_fee_percent = serializers.IntegerField( required=False, help_text="The fee for the trade (fees differ for maker and taker)" ) bond_size_sats = serializers.IntegerField( required=False, help_text="The size of the bond in sats" ) bond_size_percent = serializers.IntegerField( required=False, help_text="same as `bond_size`" ) maker_summary = SummarySerializer(required=False) taker_summary = SummarySerializer(required=False) platform_summary = PlatformSummarySerializer(required=True) expiry_message = serializers.CharField( required=False, help_text="The reason the order expired (message associated with the `expiry_reason`)" ) num_satoshis = serializers.IntegerField( required=False, help_text="only if status = `14` (Successful Trade) and is_buyer = `true`" ) sent_satoshis = serializers.IntegerField( required=False, help_text="only if status = `14` (Successful Trade) and is_buyer = `true`" ) txid = serializers.CharField( required=False, help_text="Transaction id of the on-chain swap payout. Only if status = `14` (Successful Trade) and is_buyer = `true`" ) network = serializers.CharField( required=False, help_text="The network eg. 'testnet', 'mainnet'. Only if status = `14` (Successful Trade) and is_buyer = `true`" ) class Meta: model = Order fields = ( "id", "status", "created_at", "expires_at", "type", "currency", "amount", "has_range", "min_amount", "max_amount", "payment_method", "is_explicit", "premium", "satoshis", "bondless_taker", "maker", "taker", "escrow_duration", "total_secs_exp", "penalty", "is_maker", "is_taker", "is_participant", "maker_status", "taker_status", "price_now", "premium", "premium_percentile", "num_similar_orders", "tg_enabled", "tg_token", "tg_bot_name", "is_buyer", "is_seller", "maker_nick", "taker_nick", "status_message", "is_fiat_sent", "is_disputed", "ur_nick", "ur_nick", "maker_locked", "taker_locked", "escrow_locked", "trade_satoshis", "bond_invoice", "bond_satoshis", "escrow_invoice", "escrow_satoshis", "invoice_amount", "swap_allowed", 'swap_failure_reason', "suggested_mining_fee_rate", "swap_fee_rate", "pending_cancel", "asked_for_cancel", "statement_submitted", "retries", "next_retry_time", "failure_reason", "invoice_expired", "public_duration", "bond_size", "trade_fee_percent", "bond_size_sats", "bond_size_percent", "maker_summary", "taker_summary", "platform_summary", "expiry_reason", "expiry_message", "num_satoshis", "sent_satoshis", "txid", "network", ) class OrderPublicSerializer(serializers.ModelSerializer): maker_nick = serializers.CharField(required=False) maker_status = serializers.CharField(help_text='Status of the nick - "Active" or "Inactive"', required=False) price = serializers.FloatField(help_text="Price in order's fiat currency", required=False) satoshis_now = serializers.IntegerField(help_text="The amount of sats to be traded at the present moment (not including the fees)", required=False) class Meta: model = Order fields = ( "id", "created_at", "expires_at", "type", "currency", "amount", "has_range", "min_amount", "max_amount", "payment_method", "is_explicit", "premium", "satoshis", "bondless_taker", "maker", "maker_nick", "maker_status", "price", "escrow_duration", ) class MakeOrderSerializer(serializers.ModelSerializer): currency = serializers.IntegerField( required=True, help_text="Currency id. See [here](https://github.com/Reckless-Satoshi/robosats/blob/main/frontend/static/assets/currencies.json) for a list of all IDs", ) payment_method = serializers.CharField( max_length=70, default="not specified", required=False, help_text="Can be any string. The UI recognizes [these payment methods](https://github.com/Reckless-Satoshi/robosats/blob/main/frontend/src/components/payment-methods/Methods.js) and displays them with a logo." ) is_explicit = serializers.BooleanField( default=False, help_text='Whether the order is explicitly priced or not. If set to `true` then `satoshis` need to be specified' ) has_range = serializers.BooleanField( default=False, help_text='Whether the order specifies a range of amount or a fixed amount.\n\nIf `true`, then `min_amount` and `max_amount` fields are **required**.\n\n If `false` then `amount` is **required**', ) bondless_taker = serializers.BooleanField( default=False, help_text='Whether bondless takers are allowed for this order or not', ) class Meta: model = Order fields = ( "type", "currency", "amount", "has_range", "min_amount", "max_amount", "payment_method", "is_explicit", "premium", "satoshis", "public_duration", "escrow_duration", "bond_size", "bondless_taker", ) class UpdateOrderSerializer(serializers.Serializer): invoice = serializers.CharField(max_length=2000, allow_null=True, allow_blank=True, default=None) address = serializers.CharField(max_length=100, allow_null=True, allow_blank=True, default=None) statement = serializers.CharField(max_length=10000, allow_null=True, allow_blank=True, default=None) action = serializers.ChoiceField( choices=( "pause", "take", "update_invoice", "update_address", "submit_statement", "dispute", "cancel", "confirm", "rate_user", "rate_platform", ), allow_null=False, ) rating = serializers.ChoiceField( choices=("1", "2", "3", "4", "5"), allow_null=True, allow_blank=True, default=None, ) amount = serializers.DecimalField(max_digits=18, decimal_places=8, allow_null=True, required=False, default=None) mining_fee_rate = serializers.DecimalField(max_digits=6, decimal_places=3, allow_null=True, required=False, default=None) class UserGenSerializer(serializers.Serializer): # Mandatory fields token_sha256 = serializers.CharField( min_length=64, max_length=64, allow_null=False, allow_blank=False, required=True, help_text="SHA256 of user secret") public_key = serializers.CharField(max_length=2000, allow_null=False, allow_blank=False, required=True, help_text="Armored ASCII PGP public key block") encrypted_private_key = serializers.CharField(max_length=2000, allow_null=False, allow_blank=False, required=True, help_text="Armored ASCII PGP encrypted private key block") # Optional fields ref_code = serializers.CharField(max_length=30, allow_null=True, allow_blank=True, required=False, default=None, help_text="Referal code") counts = serializers.ListField(child=serializers.IntegerField(), allow_null=True, required=False, default=None, help_text="Counts of the unique characters in the token") length = serializers.IntegerField(allow_null=True, default=None, required=False, min_value=1, help_text="Length of the token") unique_values = serializers.IntegerField(allow_null=True, default=None, required=False, min_value=1, help_text="Number of unique values in the token") class ClaimRewardSerializer(serializers.Serializer): invoice = serializers.CharField(max_length=2000, allow_null=True, allow_blank=True, default=None, help_text="A valid LN invoice with the reward amount to withdraw") class PriceSerializer(serializers.Serializer): pass class TickSerializer(serializers.ModelSerializer): class Meta: model = MarketTick fields = ( "timestamp", "currency", "volume", "price", "premium", "fee", ) depth = 1 class StealthSerializer(serializers.Serializer): wantsStealth = serializers.BooleanField()