dero-swaps/swap_manager.go

214 lines
5.5 KiB
Go
Raw Normal View History

2024-04-11 14:35:17 +02:00
package main
import (
"encoding/json"
"fmt"
"io/fs"
"log"
"os"
2024-04-20 22:23:48 +02:00
"swap-client/coin"
"swap-client/dero"
"swap-client/monero"
2024-04-11 14:35:17 +02:00
"time"
"github.com/deroproject/derohe/rpc"
)
2024-04-20 22:21:02 +02:00
const (
SWAP_CREATED = iota
SWAP_CONFIRMED
SWAP_DONE
SWAP_EXPIRED
SWAP_TOO_OLD
SWAP_UNDEFINED
)
2024-04-11 14:35:17 +02:00
func Swap_Controller() {
var file_data []byte
var swap_e coin.Swap_Entry
var expired, fails, sent, active uint
var txs []rpc.Transfer
var xmr_txs []monero.RPC_XMR_Transfer_Params
var dir_entries []fs.DirEntry
var err error
for {
time.Sleep(time.Minute)
dir_entries, err = os.ReadDir("swaps/active")
2024-04-20 22:21:02 +02:00
if err != nil {
log.Println("Can't list swap entries")
continue
}
2024-04-11 14:35:17 +02:00
expired = 0
fails = 0
sent = 0
active = 0
txs = nil
xmr_txs = nil
for _, e := range dir_entries {
active++
file_data = nil
err = nil
file_data, err = os.ReadFile("swaps/active/" + e.Name())
if err != nil {
fails++
continue
}
err = json.Unmarshal(file_data, &swap_e)
if err != nil {
fails++
continue
}
creation_t := time.UnixMilli(swap_e.Created)
// if there was no deposit, mark the request as expired
2024-04-20 22:21:02 +02:00
if swap_e.Status == SWAP_CREATED && time.Since(creation_t) > time.Hour {
2024-04-11 14:35:17 +02:00
os.WriteFile(fmt.Sprintf("swaps/expired/%d", swap_e.Created), file_data, 0644)
os.Remove("swaps/active/" + e.Name())
switch swap_e.Coin {
case coin.LTCDERO, coin.BTCDERO, coin.ARRRDERO, coin.XMRDERO:
coin.Locked.RemoveLockedBalance(swap_e.Coin, swap_e.Amount)
default:
coin.Locked.RemoveLockedBalance(swap_e.Coin, swap_e.Price)
}
expired++
continue
}
var found_deposit, visible bool
// check for deposits
switch swap_e.Coin {
case coin.BTCDERO, coin.LTCDERO, coin.ARRRDERO:
found_deposit, visible, _, err = coin.XTCListReceivedByAddress(swap_e.Coin, swap_e.Wallet, swap_e.Price, swap_e.Block, false)
case coin.XMRDERO:
if payment_id := monero.SplitIntegratedAddress(swap_e.Wallet); payment_id != "" {
found_deposit = monero.XMRGetTX(payment_id, swap_e.Block)
visible = found_deposit
} else {
2024-04-20 22:21:02 +02:00
log.Println("Can't split integrated XMR address")
2024-04-11 14:35:17 +02:00
}
default:
found_deposit = dero.CheckIncomingTransfers(uint64(swap_e.Created), swap_e.Block)
visible = found_deposit
}
if err != nil {
log.Printf("Error checking incoming %s transactions\n", swap_e.Coin)
fails++
continue
}
// mark request as done
2024-04-20 22:21:02 +02:00
if swap_e.Status == SWAP_DONE {
2024-04-11 14:35:17 +02:00
err = os.WriteFile(fmt.Sprintf("swaps/done/%d", swap_e.Created), file_data, 0644)
if err != nil {
log.Printf("Can't mark swap as done, swap %d, err %v\n", swap_e.Created, err)
} else {
os.Remove("swaps/active/" + e.Name())
}
}
// start payout if there are at least 2 confirmations
// requests won't be marked as expired, if there is already 1 confirmation
if visible {
2024-04-20 22:21:02 +02:00
log.Printf("Found TX for ID %d (%s) on chain\n", swap_e.Created, swap_e.Coin)
if found_deposit && swap_e.Status <= SWAP_CONFIRMED {
2024-04-11 14:35:17 +02:00
// create transaction
log.Printf("Found deposit for ID %d (%s): %.8f coins; adding to payout TX\n", swap_e.Created, swap_e.Coin, swap_e.Amount)
switch swap_e.Coin {
case coin.DEROLTC, coin.DEROBTC:
log.Println("Starting LTC/BTC payout")
_, txid := coin.XTCSend(swap_e.Coin, swap_e.Destination, swap_e.Price)
2024-04-20 22:21:02 +02:00
log.Printf("LTC/BTC TXID: %s\n", txid)
2024-04-11 14:35:17 +02:00
coin.Locked.RemoveLockedBalance(swap_e.Coin, swap_e.Price)
case coin.DEROARRR:
log.Println("Starting ARRR payout")
ok, result := coin.ARRR_Send(swap_e.Destination, swap_e.Price)
log.Printf("ARRR status: %v, %s\n", ok, result)
coin.Locked.RemoveLockedBalance(swap_e.Coin, swap_e.Price)
case coin.DEROXMR:
xmr_txs = append(xmr_txs, monero.AddTX(swap_e.Destination, swap_e.Price))
default:
txs = append(txs, dero.AddTX(swap_e.Destination, swap_e.Amount))
}
2024-04-20 22:21:02 +02:00
swap_e.Status = SWAP_DONE
2024-04-11 14:35:17 +02:00
sent++
active--
} else {
// transaction was confirmed
2024-04-20 22:21:02 +02:00
swap_e.Status = SWAP_CONFIRMED
2024-04-11 14:35:17 +02:00
}
json_data, _ := json.Marshal(&swap_e)
os.WriteFile("swaps/active/"+e.Name(), json_data, 0644)
2024-04-20 22:21:02 +02:00
if swap_e.Status == SWAP_DONE {
2024-04-11 14:35:17 +02:00
err = os.WriteFile(fmt.Sprintf("swaps/done/%d", swap_e.Created), file_data, 0644)
if err != nil {
log.Printf("Can't mark swap as done, swap %d, err %v\n", swap_e.Created, err)
} else {
os.Remove("swaps/active/" + e.Name())
}
}
}
}
// Dero and Monero payout process
if len(txs) > 0 {
log.Println("Starting DERO payout process")
dero.Payout(txs)
}
// TODO: create function and TX verification
if len(xmr_txs) > 0 {
log.Println("Starting XMR payout process")
if ok, txid := monero.XMRSend(xmr_txs); ok {
log.Printf("XMR transaction (TXID %s) successfully sent\n", txid)
} else {
log.Println("Error sending XMR transaction")
}
}
if sent+expired+fails > 0 {
log.Printf("Swap processing: %d sent, %d expired, %d errors\n", sent, expired, fails)
}
}
}
2024-04-20 22:21:02 +02:00
func SwapTracking(session int64) uint64 {
if time.Since(time.UnixMilli(int64(session))) > time.Hour*24*3 {
return SWAP_TOO_OLD
}
if _, err := os.ReadFile("swaps/expired/" + fmt.Sprintf("%d", session)); err == nil {
return SWAP_EXPIRED
}
if _, err := os.ReadFile("swaps/done/" + fmt.Sprintf("%d", session)); err == nil {
return SWAP_DONE
}
var swap coin.Swap_Entry
if fd, err := os.ReadFile("swaps/active/" + fmt.Sprintf("%d", session)); err == nil {
if err = json.Unmarshal(fd, &swap); err == nil {
return swap.Status
} else {
log.Println(err)
}
} else {
log.Println(err)
}
return SWAP_UNDEFINED
}