124 lines
4.6 KiB
C#
124 lines
4.6 KiB
C#
using Logging;
|
|
using Nethereum.ABI.FunctionEncoding.Attributes;
|
|
using Nethereum.Contracts;
|
|
using Nethereum.RPC.Eth.DTOs;
|
|
using Nethereum.Web3;
|
|
using NethereumWorkflow.BlockUtils;
|
|
using Utils;
|
|
|
|
namespace NethereumWorkflow
|
|
{
|
|
public class NethereumInteraction
|
|
{
|
|
// BlockCache is a static instance: It stays alive for the duration of the application runtime.
|
|
private readonly static BlockCache blockCache = new BlockCache();
|
|
|
|
private readonly ILog log;
|
|
private readonly Web3 web3;
|
|
|
|
internal NethereumInteraction(ILog log, Web3 web3)
|
|
{
|
|
this.log = log;
|
|
this.web3 = web3;
|
|
}
|
|
|
|
public string SendEth(string toAddress, decimal ethAmount)
|
|
{
|
|
log.Debug();
|
|
var receipt = Time.Wait(web3.Eth.GetEtherTransferService().TransferEtherAndWaitForReceiptAsync(toAddress, ethAmount));
|
|
if (!receipt.Succeeded()) throw new Exception("Unable to send Eth");
|
|
return receipt.TransactionHash;
|
|
}
|
|
|
|
public decimal GetEthBalance()
|
|
{
|
|
log.Debug();
|
|
return GetEthBalance(web3.TransactionManager.Account.Address);
|
|
}
|
|
|
|
public decimal GetEthBalance(string address)
|
|
{
|
|
log.Debug();
|
|
var balance = Time.Wait(web3.Eth.GetBalance.SendRequestAsync(address));
|
|
return Web3.Convert.FromWei(balance.Value);
|
|
}
|
|
|
|
public TResult Call<TFunction, TResult>(string contractAddress, TFunction function) where TFunction : FunctionMessage, new()
|
|
{
|
|
log.Debug(typeof(TFunction).ToString());
|
|
var handler = web3.Eth.GetContractQueryHandler<TFunction>();
|
|
return Time.Wait(handler.QueryAsync<TResult>(contractAddress, function));
|
|
}
|
|
|
|
public string SendTransaction<TFunction>(string contractAddress, TFunction function) where TFunction : FunctionMessage, new()
|
|
{
|
|
log.Debug();
|
|
var handler = web3.Eth.GetContractTransactionHandler<TFunction>();
|
|
var receipt = Time.Wait(handler.SendRequestAndWaitForReceiptAsync(contractAddress, function));
|
|
if (!receipt.Succeeded()) throw new Exception("Unable to perform contract transaction.");
|
|
return receipt.TransactionHash;
|
|
}
|
|
|
|
public Transaction GetTransaction(string transactionHash)
|
|
{
|
|
log.Debug();
|
|
return Time.Wait(web3.Eth.Transactions.GetTransactionByHash.SendRequestAsync(transactionHash));
|
|
}
|
|
|
|
public decimal? GetSyncedBlockNumber()
|
|
{
|
|
log.Debug();
|
|
var sync = Time.Wait(web3.Eth.Syncing.SendRequestAsync());
|
|
var number = Time.Wait(web3.Eth.Blocks.GetBlockNumber.SendRequestAsync());
|
|
var numberOfBlocks = number.ToDecimal();
|
|
if (sync.IsSyncing) return null;
|
|
return numberOfBlocks;
|
|
}
|
|
|
|
public bool IsContractAvailable(string abi, string contractAddress)
|
|
{
|
|
log.Debug();
|
|
try
|
|
{
|
|
var contract = web3.Eth.GetContract(abi, contractAddress);
|
|
return contract != null;
|
|
}
|
|
catch
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public List<EventLog<TEvent>> GetEvents<TEvent>(string address, TimeRange timeRange) where TEvent : IEventDTO, new()
|
|
{
|
|
var wrapper = new Web3Wrapper(web3, log);
|
|
var blockTimeFinder = new BlockTimeFinder(blockCache, wrapper, log);
|
|
|
|
var fromBlock = blockTimeFinder.GetLowestBlockNumberAfter(timeRange.From);
|
|
var toBlock = blockTimeFinder.GetHighestBlockNumberBefore(timeRange.To);
|
|
|
|
if (!fromBlock.HasValue)
|
|
{
|
|
log.Error("Failed to find lowest block for time range: " + timeRange);
|
|
throw new Exception("Failed");
|
|
}
|
|
if (!toBlock.HasValue)
|
|
{
|
|
log.Error("Failed to find highest block for time range: " + timeRange);
|
|
throw new Exception("Failed");
|
|
}
|
|
|
|
return GetEvents<TEvent>(address, fromBlock.Value, toBlock.Value);
|
|
}
|
|
|
|
public List<EventLog<TEvent>> GetEvents<TEvent>(string address, ulong fromBlockNumber, ulong toBlockNumber) where TEvent : IEventDTO, new()
|
|
{
|
|
var eventHandler = web3.Eth.GetEvent<TEvent>(address);
|
|
var from = new BlockParameter(fromBlockNumber);
|
|
var to = new BlockParameter(toBlockNumber);
|
|
var blockFilter = Time.Wait(eventHandler.CreateFilterBlockRangeAsync(from, to));
|
|
return Time.Wait(eventHandler.GetAllChangesAsync(blockFilter));
|
|
}
|
|
}
|
|
}
|