Files
atomic-swap/protocol/xmrmaker/instance.go
2023-06-19 14:47:27 -05:00

301 lines
7.7 KiB
Go

// Copyright 2023 The AthanorLabs/atomic-swap Authors
// SPDX-License-Identifier: LGPL-3.0-only
package xmrmaker
import (
"fmt"
"sync"
"github.com/MarinX/monerorpc/wallet"
"github.com/athanorlabs/atomic-swap/coins"
"github.com/athanorlabs/atomic-swap/common"
"github.com/athanorlabs/atomic-swap/common/types"
mcrypto "github.com/athanorlabs/atomic-swap/crypto/monero"
pcommon "github.com/athanorlabs/atomic-swap/protocol"
"github.com/athanorlabs/atomic-swap/protocol/backend"
"github.com/athanorlabs/atomic-swap/protocol/swap"
"github.com/athanorlabs/atomic-swap/protocol/xmrmaker/offers"
logging "github.com/ipfs/go-log/v2"
)
var (
log = logging.Logger("xmrmaker")
)
// Host contains required network functionality.
type Host interface {
Advertise()
}
// Instance implements the functionality that will be needed by a user who owns XMR
// and wishes to swap for ETH.
type Instance struct {
backend backend.Backend
dataDir string
net Host
offerManager *offers.Manager
swapMu sync.Mutex // synchronises access to swapStates
swapStates map[types.Hash]*swapState
}
// Config contains the configuration values for a new XMRMaker instance.
type Config struct {
Backend backend.Backend
Database offers.Database
DataDir string
WalletFile, WalletPassword string
ExternalSender bool
Network Host
}
// NewInstance returns a new *xmrmaker.Instance.
// It accepts an endpoint to a monero-wallet-rpc instance where account 0 contains XMRMaker's XMR.
func NewInstance(cfg *Config) (*Instance, error) {
om, err := offers.NewManager(cfg.DataDir, cfg.Database)
if err != nil {
return nil, err
}
if om.NumOffers() > 0 {
// this is blocking if the network service hasn't started yet
go cfg.Network.Advertise()
}
inst := &Instance{
backend: cfg.Backend,
dataDir: cfg.DataDir,
offerManager: om,
swapStates: make(map[types.Hash]*swapState),
net: cfg.Network,
}
err = inst.checkForOngoingSwaps()
if err != nil {
return nil, err
}
return inst, nil
}
func (inst *Instance) checkForOngoingSwaps() error {
ongoingIDs, err := inst.backend.SwapManager().GetOngoingSwapOfferIDs()
if err != nil {
return err
}
for _, offerID := range ongoingIDs {
s, err := inst.backend.SwapManager().GetOngoingSwap(*offerID)
if err != nil {
return err
}
if s.Provides != coins.ProvidesXMR {
continue
}
if s.Status == types.KeysExchanged || s.Status == types.ExpectingKeys {
log.Infof("found ongoing swap %s in DB, aborting since no funds were locked", s.OfferID)
// for these two cases, no funds have been locked, so we can safely
// abort the swap.
err = inst.abortOngoingSwap(s)
if err != nil {
return fmt.Errorf("failed to abort ongoing swap: %w", err)
}
continue
}
if s.Status == types.SweepingXMR {
log.Infof(
"found ongoing swap %s in DB where XMR was being swept back to the primary account, marking as completed",
s.OfferID,
)
s.Status = types.CompletedRefund
err = inst.backend.SwapManager().CompleteOngoingSwap(s)
if err != nil {
return fmt.Errorf("failed to mark swap as completed: %w", err)
}
continue
}
err = inst.createOngoingSwap(s)
if err != nil {
log.Errorf("%s", err)
continue
}
}
return nil
}
func (inst *Instance) abortOngoingSwap(s *swap.Info) error {
// set status to aborted, delete info from recovery db
s.Status = types.CompletedAbort
err := inst.backend.SwapManager().CompleteOngoingSwap(s)
if err != nil {
return err
}
return inst.backend.RecoveryDB().DeleteSwap(s.OfferID)
}
func (inst *Instance) createOngoingSwap(s *swap.Info) error {
log.Infof("found ongoing swap %s in DB, restarting swap", s.OfferID)
// check if we have shared secret key in db; if so, recover XMR from that
// otherwise, create new swap state from recovery info
skA, err := inst.backend.RecoveryDB().GetCounterpartySwapPrivateKey(s.OfferID)
if err == nil {
return inst.completeSwap(s, skA)
}
offer, _, err := inst.offerManager.TakeOffer(s.OfferID)
if err != nil {
return fmt.Errorf("failed to get offer for ongoing swap, offer ID %s: %s", s.OfferID, err)
}
ethSwapInfo, err := inst.backend.RecoveryDB().GetContractSwapInfo(s.OfferID)
if err != nil {
return fmt.Errorf("failed to get contract info for ongoing swap from db with offer ID %s: %s",
s.OfferID, err)
}
sk, err := inst.backend.RecoveryDB().GetSwapPrivateKey(s.OfferID)
if err != nil {
return fmt.Errorf("failed to get private key for ongoing swap from db with offer ID %s: %s",
s.OfferID, err)
}
kp, err := sk.AsPrivateKeyPair()
if err != nil {
return err
}
relayerInfo, err := inst.backend.RecoveryDB().GetSwapRelayerInfo(s.OfferID)
if err != nil {
// we can ignore the error; if the key doesn't exist,
// then no relayer was set for this swap.
relayerInfo = types.NewOfferExtra(false)
}
ss, err := newSwapStateFromOngoing(
inst.backend,
offer,
relayerInfo,
inst.offerManager,
ethSwapInfo,
s,
kp,
)
if err != nil {
return fmt.Errorf("failed to create new swap state for ongoing swap, offer id %s: %w", s.OfferID, err)
}
inst.swapMu.Lock()
inst.swapStates[s.OfferID] = ss
inst.swapMu.Unlock()
go func() {
<-ss.done
inst.swapMu.Lock()
defer inst.swapMu.Unlock()
delete(inst.swapStates, offer.ID)
}()
return nil
}
// completeSwap is called in the case where we find an ongoing swap in the db on startup,
// and the swap already has the counterparty's swap secret stored.
// In this case, we simply re-claim the XMR we locked, as we have both secrets required.
// It's unlikely for this case to ever be hit, unless the daemon was shut down in-between
// us finding the counterparty's secret and claiming the XMR.
//
// Note: this will use the current value of `noTransferBack` (verses whatever value was
// set when the swap was started). It will also only only recover to the primary wallet
// address, not whatever address was used when the swap was started.
func (inst *Instance) completeSwap(s *swap.Info, skA *mcrypto.PrivateSpendKey) error {
// fetch our swap private spend key
skB, err := inst.backend.RecoveryDB().GetSwapPrivateKey(s.OfferID)
if err != nil {
return err
}
// fetch our swap private view key
vkB, err := skB.View()
if err != nil {
return err
}
// we save the counterparty's public keys in case they send public keys derived
// in a non-standard way.
_, vkA, err := inst.backend.RecoveryDB().GetCounterpartySwapKeys(s.OfferID)
if err != nil {
return err
}
kpAB := pcommon.GetClaimKeypair(
skA, skB,
vkA, vkB,
)
err = pcommon.ClaimMonero(
inst.backend.Ctx(),
inst.backend.Env(),
s,
inst.backend.XMRClient(),
kpAB,
inst.backend.XMRClient().PrimaryAddress(),
false, // always sweep back to our primary address
inst.backend.SwapManager(),
)
if err != nil {
return err
}
s.Status = types.CompletedRefund
err = inst.backend.SwapManager().CompleteOngoingSwap(s)
if err != nil {
return fmt.Errorf("failed to mark swap %s as completed: %w", s.OfferID, err)
}
return nil
}
// GetOngoingSwapState ...
func (inst *Instance) GetOngoingSwapState(id types.Hash) common.SwapState {
inst.swapMu.Lock()
defer inst.swapMu.Unlock()
return inst.swapStates[id]
}
// GetMoneroBalance returns the primary wallet address, and current balance of the user's monero
// wallet.
func (inst *Instance) GetMoneroBalance() (*mcrypto.Address, *wallet.GetBalanceResponse, error) {
addrResp, err := inst.backend.XMRClient().GetAddress(0)
if err != nil {
return nil, nil, err
}
addr, err := mcrypto.NewAddress(addrResp.Address, inst.backend.Env())
if err != nil {
return nil, nil, err
}
balanceResp, err := inst.backend.XMRClient().GetBalance(0)
if err != nil {
return nil, nil, err
}
return addr, balanceResp, nil
}