Nethereum.Unity 库是一个特定于 NETHereum 的 Unity 库和 API,它支持 UnityWebRequest 使用 RPC over Http 与 Ethereum 交互。Nethereum.Unity 库是唯一支持在 Unity 中使用协程时使用 IEnumerator 和 yield 的库。
用自己熟悉的语言学习 以太坊开发 : Java | Php | Python | .Net / C# | Golang | Node.JS | Flutter / Dart
1、Nethereum.Unity简介
如果想在 net461 / netstandard 中使用 async / await 和 Tasks,只要你的环境不需要使用 UnityWebRequest 而不是 HttpRequest,也可以使用与“vanilla” Nethereum 相同的方式。 (Webgl 需要使用 UnityWebRequest)
Nethereum 还为 net351 和 net461 框架构建提供了 AoT 库。 所有“dll”都可以从 Nethereum github 下载。
flappy eth 是 Unity3d 的“flappy”示例,使用 Nethereum 作为 webgl dapp 游戏转换为 与 Ethereum、Infura 和 Metamask 交互。主要集成组件的源代码可以在这里找到, 你也可以在这里尝试游戏。
2、Unity3dSimpleSample示例简介
Unity3dSimpleSample是使用 Net472(现在从 Net461 升级)进行 Unity3d 开发的简单示例,使用的 Unity 版本是 2020.3.15f LTS。 示例中包含资产文件夹中的所有 DLL,你可能不需要其中一些代码,直接删除它们(如 Besu、Geth、HdWallet、NBitcoin 等)就可以了,具体 取决于你的需要。
示例代码演示了如何使用Nethereum.Unity实现以下功能:
- 在异步和协程中使用 Unity.UI 将当前 BlockNumber 输出到日志
- 使用 Unity.UI 和协程进行以太传输
- ETH转账时使用 1559 Suggestion 策略或 Legacy 模式
- 智能合约部署(ERC20)、交易(Transfer)和查询(Balance)
注意:
WebGL 仅支持协程 UnityWebRequest。如果构建 WebGL版本发生问题,请取消选中 Development Build。为了支持 WebGL 和 AOT,此示例使用 Net472AOT dll 和自定义 Json.Net Unity请记住删除 Nethereum 发布包的 System.HttpClient 和 UnityEngine
下面是桌面版的截图:
下面是WebGL版的截图:
要运行本地区块链,可以使用预配置的测试链。
3、异步查询区块号 — 非协程方式
下面的代码展示了如何用Nethereum.Unity异步查询区块号:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
|
using System;
using System.Collections;
using System.Collections.Generic;
using System.Net;
using System.Net.Security;
using System.Security.Cryptography.X509Certificates;
using Nethereum.ABI.FunctionEncoding.Attributes;
using UnityEngine;
using UnityEngine.UI;
using Nethereum.Contracts;
using Nethereum.Web3;
public class GetLatestBlockVanillaNethereum : MonoBehaviour {
private static bool TrustCertificate(object sender, X509Certificate x509Certificate, X509Chain x509Chain, SslPolicyErrors sslPolicyErrors)
{
// all certificates are accepted
return true;
}
public string Url = "https://mainnet.infura.io";
public InputField ResultBlockNumber;
public InputField InputUrl;
// Use this for initialization
void Start()
{
InputUrl.text = Url;
}
public async void GetBlockNumber()
{
Url = InputUrl.text;
//This is to workaround issue with certificates https://forum.unity.com/threads/how-to-allow-self-signed-certificate.522183/
//Uncomment if needed
// ServicePointManager.ServerCertificateValidationCallback = TrustCertificate;
var web3 = new Web3(Url);
var blockNumber = await web3.Eth.Blocks.GetBlockNumber.SendRequestAsync();
ResultBlockNumber.text = blockNumber.Value.ToString();
}
}
|
4、异步查询区块号 — 协程方式
下面的代码展示了如何用Nethereum.Unity以协程方式异步查询区块号:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
using Nethereum.JsonRpc.UnityClient;
using Nethereum.RPC.Eth.DTOs;
using Nethereum.Util;
public class GetLatestBlockCoroutine : MonoBehaviour
{
public string Url = "https://mainnet.infura.io";
public InputField ResultBlockNumber;
public InputField InputUrl;
// Use this for initialization
void Start()
{
InputUrl.text = Url;
}
public void GetBlockNumberRequest()
{
StartCoroutine(GetBlockNumber());
}
public IEnumerator GetBlockNumber()
{
Url = InputUrl.text;
var blockNumberRequest = new EthBlockNumberUnityRequest(Url);
yield return blockNumberRequest.SendRequest();
ResultBlockNumber.text = blockNumberRequest.Result.Value.ToString();
}
}
|
5、ETH转账的简单实现
为了支持ETH转账, Nethereum 提供了一个特定的 Unity 请求,EthTransferUnityRequest.
EthTransferUnityRequest 使用我们的以太坊客户端的“url”、能够签署交易的私钥和我们的帐户 地址(与私钥相同)进行实例化。
1
2
3
|
var url = " http://localhost:8545 " ;
var privateKey = " 0xb5b1870957d373ef0eeffecc6e4812c0fd08f554b37b233526acc331bf1544f7 " ;
var ethTransfer = new EthTransferUnityRequest ( url , privateKey , " YOURCHAINID " );
|
一旦我们的请求被实例化,就可以使用传统模式启动传输,下面的代码设置 2 Gwei 作为gas价格:
1
2
|
var receivingAddress = "0xde0B295669a9FD93d5F28D9Ec85E40f4cb697BAe";
yield return ethTransfer.TransferEther(receivingAddress, 1.1m, 2);
|
在这里,我们指定了接收地址、发送量和可选的 gas 价格。该请求将自动将 gas 价格转换为 Wei。
我们可以在之后验证是否有任何异常,如下所示:
1
2
3
4
5
|
if (ethTransfer.Exception != null)
{
Debug.Log(ethTransfer.Exception.Message);
yield break;
}
|
如果没有发生错误,我们可以每 2 秒从请求和轮询中检索交易哈希以等待交易确认:
1
2
3
4
5
|
var transactionHash = ethTransfer.Result;
//create a poll to get the receipt when mined
var transactionReceiptPolling = new TransactionReceiptPollingRequest(url);
//checking every 2 seconds for the receipt
yield return transactionReceiptPolling.PollForReceipt(transactionHash, 2);
|
最后,我们可以使用EthGetBalanceUnityRequest检查收款账户的余额。请注意,我们在执行 请求时声明想要最新块记录的余额。
1
2
|
var balanceRequest = new EthGetBalanceUnityRequest(url);
yield return balanceRequest.SendRequest(receivingAddress, BlockParameter.CreateLatest());
|
可以使用默认的 Wei UnitConvertor 将 Wei 中的结果转换为 Eth:
1
|
Debug.Log("Balance of account:" UnitConversion.Convert.FromWei(balanceRequest.Result.Value));
|
6、EIP 1559 使用建议
以下是使用协程、TimePreference、MedianFeeHistory 或 LegacyMode 提供的费用建议策略以使用 旧模式或与其他链一起使用的一些示例。
要使用 LegacyMode,你必须提供 GasPrice,或者可以通过设置UseLegacyAsDefault为 true 来强制 LegacyMode:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
|
if (feeStrategy == FeeStrategy.TimePreference)
{
Debug.Log("Time Preference");
var timePreferenceFeeSuggestion = new TimePreferenceFeeSuggestionUnityRequestStrategy(Url);
yield return timePreferenceFeeSuggestion.SuggestFees();
if (timePreferenceFeeSuggestion.Exception != null)
{
Debug.Log(timePreferenceFeeSuggestion.Exception.Message);
yield break;
}
//lets get the first one so it is higher priority
Debug.Log(timePreferenceFeeSuggestion.Result.Length);
if (timePreferenceFeeSuggestion.Result.Length > 0)
{
Debug.Log(timePreferenceFeeSuggestion.Result[0].MaxFeePerGas);
Debug.Log(timePreferenceFeeSuggestion.Result[0].MaxPriorityFeePerGas);
}
var fee = timePreferenceFeeSuggestion.Result[0];
yield return ethTransfer.TransferEther(receivingAddress, Amount, fee.MaxPriorityFeePerGas.Value, fee.MaxFeePerGas.Value);
if (ethTransfer.Exception != null)
{
Debug.Log(ethTransfer.Exception.Message);
yield break;
}
}
if(feeStrategy == FeeStrategy.MedianFeeHistory)
{
Debug.Log("MedianFeeHistory mode");
var medianPriorityFeeStrategy = new MedianPriorityFeeHistorySuggestionUnityRequestStrategy(Url);
yield return medianPriorityFeeStrategy.SuggestFee();
if (medianPriorityFeeStrategy.Exception != null)
{
Debug.Log(medianPriorityFeeStrategy.Exception.Message);
yield break;
}
Debug.Log(medianPriorityFeeStrategy.Result.MaxFeePerGas);
Debug.Log(medianPriorityFeeStrategy.Result.MaxPriorityFeePerGas);
var fee = medianPriorityFeeStrategy.Result;
yield return ethTransfer.TransferEther(receivingAddress, Amount, fee.MaxPriorityFeePerGas.Value, fee.MaxFeePerGas.Value);
if (ethTransfer.Exception != null)
{
Debug.Log(ethTransfer.Exception.Message);
yield break;
}
}
if (feeStrategy == FeeStrategy.Legacy)
{
Debug.Log("Legacy mode");
//I am forcing the legacy mode but also I am including the gas price
ethTransfer.UseLegacyAsDefault = true;
yield return ethTransfer.TransferEther(receivingAddress, Amount, GasPriceGwei);
if (ethTransfer.Exception != null)
{
Debug.Log(ethTransfer.Exception.Message);
yield break;
}
}
|
7、智能合约定义声明
在游戏中实现智能合约支持的第一步是包含我们的智能合约定义,这可以是使用 vscode solidity 扩展 或控制台代码生成工具生成的代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
|
//Deployment contract object definition
public partial class EIP20Deployment : EIP20DeploymentBase
{
public EIP20Deployment() : base(byteCODE) { }
public EIP20Deployment(string byteCode) : base(byteCode) { }
}
public class EIP20DeploymentBase : ContractDeploymentMessage
{
public static string BYTECODE = "608060405234801561001057600080fd5b506040516107843803806107848339810160409081528151602080840151838501516060860151336000908152808552959095208490556002849055908501805193959094919391019161006991600391860190610096565b506004805460ff191660ff8416179055805161008c906005906020840190610096565b5050505050610131565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f106100d757805160ff1916838001178555610104565b82800160010185558215610104579182015b828111156101045782518255916020019190600101906100e9565b50610110929150610114565b5090565b61012e91905b80821115610110576000815560010161011a565b90565b610644806101406000396000f3006080604052600436106100ae5763ffffffff7c010000000000000000000000000000000000000000000000000000000060003504166306fdde0381146100b3578063095ea7b31461013d57806318160ddd1461017557806323b872dd1461019c57806327e235e3146101c6578063313ce567146101e75780635c6581651461021257806370a082311461023957806395d89b411461025a578063a9059cbb1461026f578063dd62ed3e14610293575b600080fd5b3480156100bf57600080fd5b506100c86102ba565b6040805160208082528351818301528351919283929083019185019080838360005b838110156101025781810151838201526020016100ea565b50505050905090810190601f16801561012f5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b34801561014957600080fd5b50610161600160a060020a0360043516602435610348565b604080519115158252519081900360200190f35b34801561018157600080fd5b5061018a6103ae565b60408051918252519081900360200190f35b3480156101a857600080fd5b50610161600160a060020a03600435811690602435166044356103b4565b3480156101d257600080fd5b5061018a600160a060020a03600435166104b7565b3480156101f357600080fd5b506101fc6104c9565b6040805160ff9092168252519081900360200190f35b34801561021e57600080fd5b5061018a600160a060020a03600435811690602435166104d2565b34801561024557600080fd5b5061018a600160a060020a03600435166104ef565b34801561026657600080fd5b506100c861050a565b34801561027b57600080fd5b50610161600160a060020a0360043516602435610565565b34801561029f57600080fd5b5061018a600160a060020a03600435811690602435166105ed565b6003805460408051602060026001851615610100026000190190941693909304601f810184900484028201840190925281815292918301828280156103405780601f1061031557610100808354040283529160200191610340565b820191906000526020600020905b81548152906001019060200180831161032357829003601f168201915b505050505081565b336000818152600160209081526040808320600160a060020a038716808552908352818420869055815186815291519394909390927f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925928290030190a350600192915050565b60025481565b600160a060020a03831660008181526001602090815260408083203384528252808320549383529082905281205490919083118015906103f45750828110155b15156103ff57600080fd5b600160a060020a038085166000908152602081905260408082208054870190559187168152208054849003905560001981101561046157600160a060020a03851660009081526001602090815260408083203384529091529020805484900390555b83600160a060020a031685600160a060020a03167fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef856040518082815260200191505060405180910390a3506001949350505050565b60006020819052908152604090205481565b60045460ff1681565b600160209081526000928352604080842090915290825290205481565b600160a060020a031660009081526020819052604090205490565b6005805460408051602060026001851615610100026000190190941693909304601f810184900484028201840190925281815292918301828280156103405780601f1061031557610100808354040283529160200191610340565b3360009081526020819052604081205482111561058157600080fd5b3360008181526020818152604080832080548790039055600160a060020a03871680845292819020805487019055805186815290519293927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef929181900390910190a350600192915050565b600160a060020a039182166000908152600160209081526040808320939094168252919091522054905600a165627a7a7230582084c618322109054a21a57e27075384a6172ab854e4b2c2d35062a964a6bf593f0029";
public EIP20DeploymentBase() : base(BYTECODE) { }
public EIP20DeploymentBase(string byteCode) : base(byteCode) { }
[Parameter("uint256", "_initialAmount", 1)]
public BigInteger InitialAmount { get; set; }
[Parameter("string", "_tokenName", 2)]
public string TokenName { get; set; }
[Parameter("uint8", "_decimalUnits", 3)]
public byte DecimalUnits { get; set; }
[Parameter("string", "_tokenSymbol", 4)]
public string TokenSymbol { get; set; }
}
[Function("transfer", "bool")]
public class TransferFunctionBase : FunctionMessage
{
[Parameter("address", "_to", 1)]
public string To { get; set; }
[Parameter("uint256", "_value", 2)]
public BigInteger Value { get; set; }
}
public partial class TransferFunction : TransferFunctionBase
{
}
[Function("balanceOf", "uint256")]
public class BalanceOfFunction : FunctionMessage
{
[Parameter("address", "_owner", 1)]
public string Owner { get; set; }
}
[FunctionOutput]
public class BalanceOfFunctionOutput : IFunctionOutputDTO
{
[Parameter("uint256", 1)]
public int Balance { get; set; }
}
[Event("Transfer")]
public class TransferEventDTOBase : IEventDTO
{
[Parameter("address", "_from", 1, true)]
public virtual string From { get; set; }
[Parameter("address", "_to", 2, true)]
public virtual string To { get; set; }
[Parameter("uint256", "_value", 3, false)]
public virtual BigInteger Value { get; set; }
}
public partial class TransferEventDTO : TransferEventDTOBase
{
public static EventABI GetEventABI()
{
return EventExtensions.GetEventABI<TransferEventDTO>();
}
}
|
8、智能合约部署
为了部署智能合约,我们使用节点 url 和签名信息创建一个 TransactionSignedUnityRequest。 创建一个新的 EIP20Deployment 合约定义,我们设置构造函数参数并发送交易。最后,我们创建 TransactionReceiptPollingRequest 来轮询交易收据并从交易收据中检索新部署的合约地址:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
var url = "http://localhost:8545";
var privateKey = "0xb5b1870957d373ef0eeffecc6e4812c0fd08f554b37b233526acc331bf1544f7";
var account = "0x12890d2cce102216644c59daE5baed380d84830c";
//initialising the transaction request sender
var transactionRequest = new TransactionSignedUnityRequest(url, privateKey, "YOURCHAINID");
var deployContract = new EIP20Deployment()
{
InitialAmount = 10000,
FromAddress = account,
TokenName = "TST",
TokenSymbol = "TST"
};
//deploy the contract
yield return transactionRequest.SignAndSendDeploymentContractTransaction<EIP20DeploymentBase>(deployContract);
if (transactionRequest.Exception != null)
{
Debug.Log(transactionRequest.Exception.Message);
yield break;
}
var transactionHash = transactionRequest.Result;
Debug.Log("Deployment transaction hash:" transactionHash);
//create a poll to get the receipt when mined
var transactionReceiptPolling = new TransactionReceiptPollingRequest(url);
//checking every 2 seconds for the receipt
yield return transactionReceiptPolling.PollForReceipt(transactionHash, 2);
var deploymentReceipt = transactionReceiptPolling.Result;
Debug.Log("Deployment contract address:" deploymentReceipt.ContractAddress);
|
9、查询智能合约
要查询智能合约,我们需要创建一个提供 FunctionType 和 ReturnType 的新 QueryUnityRequest。 然后我们将执行查询,查询结果 Result 对象将为我们提供已经解码的合约的输出:
1
2
3
4
5
6
7
|
//Query request using our acccount and the contracts address (no parameters needed and default values)
var queryRequest = new QueryUnityRequest<BalanceOfFunction, BalanceOfFunctionOutput>(url, account);
yield return queryRequest.Query(new BalanceOfFunction(){Owner = account}, deploymentReceipt.ContractAddress);
//Getting the dto response already decoded
var dtoResult = queryRequest.Result;
Debug.Log(dtoResult.Balance);
|
10、转账交易
发送交易以便与与智能合约进行交互的步骤类似于部署。我们首先创建一个 TransactionSignedUnityRequest 和包含任何参数的函数,一旦发送交易,我们会轮询确认交易成功的交易收据。
使用交易收据,我们可以解码该交易的任何日志/事件:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
var transactionTransferRequest = new TransactionSignedUnityRequest(url, privateKey, "YOURCHAINID");
var newAddress = "0xde0B295669a9FD93d5F28D9Ec85E40f4cb697BAe";
var transactionMessage = new TransferFunction
{
FromAddress = account,
To = newAddress,
Value = 1000,
};
yield return transactionTransferRequest.SignAndSendTransaction(transactionMessage, deploymentReceipt.ContractAddress);
var transactionTransferHash = transactionTransferRequest.Result;
Debug.Log("Transfer txn hash:" transactionHash);
transactionReceiptPolling = new TransactionReceiptPollingRequest(url);
yield return transactionReceiptPolling.PollForReceipt(transactionTransferHash, 2);
var transferReceipt = transactionReceiptPolling.Result;
var transferEvent = transferReceipt.DecodeAllEvents<TransferEventDTO>();
Debug.Log("Transferd amount from event: " transferEvent[0].Event.Value);
|
11、日志和事件
为了检索智能合约的日志/事件,我们使用 EthGetLogsUnityRequest 和特定于我们事件的 FilterInput。 可以使用 EventDTO 扩展 GetEventABI() 创建 FilterInputs。一旦我们生成了请求,我们就可以使用 Result.DecodeAllEvents 扩展方法解码所有匹配的事件。
1
2
3
4
5
|
var getLogsRequest = new EthGetLogsUnityRequest(url);
var eventTransfer = TransferEventDTO.GetEventABI();
yield return getLogsRequest.SendRequest(eventTransfer.CreateFilterInput(deploymentReceipt.ContractAddress, account));
var eventDecoded = getLogsRequest.Result.DecodeAllEvents<TransferEventDTO>();
Debug.Log("Transferd amount from get logs event: " eventDecoded[0].Event.Value);
|
原文链接:http://blog.hubwiz.com/2022/01/28/nethereum-unity-tutorial/
,