Files
atomic-swap/coins/exchange_rate.go

101 lines
3.2 KiB
Go

// Copyright 2023 The AthanorLabs/atomic-swap Authors
// SPDX-License-Identifier: LGPL-3.0-only
package coins
import (
"github.com/cockroachdb/apd/v3"
)
// ExchangeRate defines an exchange rate between ETH and XMR.
// It is defined as the ratio of ETH:XMR that the node wishes to provide.
// ie. an ExchangeRate of 0.1 means that the node considers 1 ETH = 10 XMR.
type ExchangeRate apd.Decimal
// CalcExchangeRate computes and returns an exchange rate using ETH and XRM prices. The
// price can be relative to USD, bitcoin or something else, but both values should be
// relative to the same alternate currency.
func CalcExchangeRate(xmrPrice *apd.Decimal, ethPrice *apd.Decimal) (*ExchangeRate, error) {
rate := new(apd.Decimal)
_, err := decimalCtx.Quo(rate, xmrPrice, ethPrice)
if err != nil {
return nil, err
}
if err = roundToDecimalPlace(rate, rate, MaxExchangeRateDecimals); err != nil {
return nil, err
}
return ToExchangeRate(rate), nil
}
// ToExchangeRate casts an *apd.Decimal to *ExchangeRate
func ToExchangeRate(rate *apd.Decimal) *ExchangeRate {
return (*ExchangeRate)(rate)
}
// Decimal casts *ExchangeRate to *apd.Decimal
func (r *ExchangeRate) Decimal() *apd.Decimal {
return (*apd.Decimal)(r)
}
// UnmarshalText hands off JSON decoding to apd.Decimal
func (r *ExchangeRate) UnmarshalText(b []byte) error {
err := r.Decimal().UnmarshalText(b)
if err != nil {
return err
}
return ValidatePositive("exchangeRate", MaxExchangeRateDecimals, r.Decimal())
}
// MarshalText hands off JSON encoding to apd.Decimal
func (r *ExchangeRate) MarshalText() ([]byte, error) {
return r.Decimal().MarshalText()
}
// ToXMR converts an ETH amount to an XMR amount with the given exchange rate
func (r *ExchangeRate) ToXMR(ethAmount *apd.Decimal) (*apd.Decimal, error) {
xmrAmt := new(apd.Decimal)
_, err := decimalCtx.Quo(xmrAmt, ethAmount, r.Decimal())
if err != nil {
return nil, err
}
if err = roundToDecimalPlace(xmrAmt, xmrAmt, NumMoneroDecimals); err != nil {
return nil, err
}
return xmrAmt, nil
}
// ToETH converts an XMR amount to an ETH amount with the given exchange rate
func (r *ExchangeRate) ToETH(xmrAmount *apd.Decimal) (*apd.Decimal, error) {
ethAmt := new(apd.Decimal)
_, err := decimalCtx.Mul(ethAmt, r.Decimal(), xmrAmount)
if err != nil {
return nil, err
}
// Assuming the xmrAmount was capped at 12 decimal places and the exchange
// rate was capped at 6 decimal places, you can't generate more than 18
// decimal places below, so no rounding occurs.
if err = roundToDecimalPlace(ethAmt, ethAmt, NumEtherDecimals); err != nil {
return nil, err
}
return ethAmt, nil
}
// ToERC20Amount converts an XMR amount to a token amount in standard units with
// the given exchange rate
func (r *ExchangeRate) ToERC20Amount(xmrAmount *apd.Decimal, token *ERC20TokenInfo) (*apd.Decimal, error) {
erc20Amount := new(apd.Decimal)
_, err := decimalCtx.Mul(erc20Amount, r.Decimal(), xmrAmount)
if err != nil {
return nil, err
}
// The token, if required, will get rounded to whole token units in
// NewERC20TokenAmountFromDecimals.
return NewERC20TokenAmountFromDecimals(erc20Amount, token).AsStandard(), nil
}
func (r *ExchangeRate) String() string {
return r.Decimal().Text('f')
}