Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add command to download a contract #209

Merged
merged 40 commits into from
Apr 19, 2022
Merged
Show file tree
Hide file tree
Changes from 33 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
b085943
Add command to download a contract from a remote network into our loc…
ixje Mar 15, 2022
f389c64
Process feedback
ixje Mar 18, 2022
4fd6638
Add ExpressFindStatesAsync without key decoding and with paging support
ixje Mar 21, 2022
5e5ad53
Merge branch 'master' into download-contract
devhawk Mar 21, 2022
061f2b4
batch command
ixje Mar 22, 2022
6799ef1
Merge master
ixje Mar 22, 2022
2a3f306
Merge branch 'download-contract' of https://github.com/ixje/neo-expre…
ixje Mar 22, 2022
f4f2382
Resolve null warnings
ixje Mar 22, 2022
d241f86
Merge branch 'master' into download-contract
ixje Mar 23, 2022
0c20e5d
handle duplicate contracts
ixje Mar 28, 2022
daa9e85
process feedback
ixje Mar 30, 2022
0e992af
Update ExpressFindStatesAsync
Mar 30, 2022
5d590c1
pass storage pairs as IReadOnlyList instead of as an array
Mar 30, 2022
c83f7b0
remove EncodedFoundStates
Mar 30, 2022
ea28151
Merge branch 'master' into pr/ixje/209
Mar 31, 2022
86ffc7d
Merge branch 'master' into download-contract
ixje Apr 5, 2022
4e59429
fix LastUsedContractId and relevant logic
ixje Apr 5, 2022
c4a51a4
Merge branch 'download-contract' of https://github.com/ixje/neo-expre…
ixje Apr 5, 2022
512cd6c
- Limit contract download to single node consensus setup
ixje Apr 7, 2022
260f123
merge master
ixje Apr 7, 2022
eacec33
check consensus node count before getting express node
Apr 7, 2022
2871ce6
get contract state via state service
Apr 7, 2022
c07ce73
fix storage persisting if local contract does not yet exist
ixje Apr 8, 2022
29520df
Merge branch 'master' into download-contract
ixje Apr 8, 2022
ee6d309
Merge branch 'download-contract' of https://github.com/ixje/neo-expre…
ixje Apr 8, 2022
30a755d
Improve error message when state service plugin is not installed
ixje Apr 12, 2022
0b0a2f4
Improve error message when contract already exists
ixje Apr 12, 2022
98baa1f
Merge branch 'master' into download-contract
ixje Apr 12, 2022
3f26fa6
Generic collections > object model collections
Apr 14, 2022
b3709c9
variable rename + format
Apr 14, 2022
b9f8658
simplify GetStateHeightAsync logic
Apr 14, 2022
7183e09
minor fixed + format
Apr 14, 2022
47f1989
reworked PersistContract
Apr 14, 2022
21c3b77
fix compile break
Apr 14, 2022
be8a346
don't create empty array via new
Apr 14, 2022
b3b6eac
reworked PersistContract
Apr 15, 2022
dcacd28
throw better exeption on missing contract + disallow native contract …
Apr 15, 2022
8b658dc
Remove unused code + touch up
ixje Apr 18, 2022
257bc39
very minor whitespace + cleanup
Apr 19, 2022
5d9f89f
update changelog
Apr 19, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
23 changes: 22 additions & 1 deletion src/neoxp/Commands/BatchCommand.BatchFileCommands.cs
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ internal class Create
}

[Command("contract")]
[Subcommand(typeof(Deploy), typeof(Invoke), typeof(Run))]
[Subcommand(typeof(Deploy), typeof(Download), typeof(Invoke), typeof(Run))]
internal class Contract
{
[Command("deploy")]
Expand All @@ -56,6 +56,27 @@ internal class Deploy
[Option(Description = "Deploy contract regardless of name conflict")]
internal bool Force { get; }
}

[Command("download")]
internal class Download
{
[Argument(0, Description = "Contract invocation hash")]
[Required]
internal string Contract { get; init; } = string.Empty;

[Argument(1, Description = "URL of Neo JSON-RPC Node\nSpecify MainNet (default), TestNet or JSON-RPC URL")]
[Required]
internal string RpcUri { get; } = string.Empty;

[Argument(2, Description = "Block height to get contract state for")]
[Required]
internal uint Height { get; } = 0;

[Option(CommandOptionType.SingleOrNoValue,
Description = "Replace contract and storage if it already exists (Default: All)")]
[AllowedValues(StringComparison.OrdinalIgnoreCase, "All", "ContractOnly", "StorageOnly")]
internal ContractCommand.OverwriteForce Force { get; init; } = ContractCommand.OverwriteForce.None;
}

[Command("invoke")]
internal class Invoke
Expand Down
24 changes: 24 additions & 0 deletions src/neoxp/Commands/BatchCommand.cs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
using System.Threading.Tasks;
using McMaster.Extensions.CommandLineUtils;
using Neo.BlockchainToolkit;
using NeoExpress.Node;

namespace NeoExpress.Commands
{
Expand Down Expand Up @@ -122,6 +123,29 @@ await txExec.ContractDeployAsync(
cmd.Model.Force).ConfigureAwait(false);
break;
}
case CommandLineApplication<BatchFileCommands.Contract.Download> cmd:
{
if (cmd.Model.Height == 0)
{
throw new ArgumentException("Height cannot be 0. Please specify a height > 0");
}

if (chainManager.Chain.ConsensusNodes.Count != 1)
{
throw new ArgumentException("Contract download is only supported for single-node consensus");
}

var expressNode = txExec.ExpressNode;
var result = await NodeUtility.DownloadContractStateAsync(
cmd.Model.Contract,
cmd.Model.RpcUri,
cmd.Model.Height).ConfigureAwait(false);
await expressNode.PersistContractAsync(
result.contractState,
result.storagePairs,
cmd.Model.Force).ConfigureAwait(false);
break;
}
case CommandLineApplication<BatchFileCommands.Contract.Invoke> cmd:
{
var script = await txExec.LoadInvocationScriptAsync(
Expand Down
77 changes: 77 additions & 0 deletions src/neoxp/Commands/ContractCommand.Download.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
using System;
using System.ComponentModel.DataAnnotations;
using System.IO;
using System.Threading.Tasks;
using McMaster.Extensions.CommandLineUtils;
using NeoExpress.Node;

namespace NeoExpress.Commands
{
partial class ContractCommand
{
internal enum OverwriteForce
{
None,
All,
ContractOnly,
StorageOnly
}

[Command(Name = "download", Description = "Download contract with storage from remote chain into local chain")]
internal class Download
{
readonly ExpressChainManagerFactory chainManagerFactory;

public Download(ExpressChainManagerFactory chainManagerFactory)
{
this.chainManagerFactory = chainManagerFactory;
}

[Argument(0, Description = "Contract invocation hash")]
[Required]
internal string Contract { get; init; } = string.Empty;

[Argument(1, Description = "URL of Neo JSON-RPC Node\nSpecify MainNet (default), TestNet or JSON-RPC URL")]
internal string RpcUri { get; } = string.Empty;

[Option(Description = "Path to neo-express data file")]
internal string Input { get; init; } = string.Empty;

[Option(Description = "Block height to get contract state for")]
internal uint Height { get; } = 0;

[Option(CommandOptionType.SingleOrNoValue,
Description = "Replace contract and storage if it already exists (Default: All)")]
[AllowedValues(StringComparison.OrdinalIgnoreCase, "All", "ContractOnly", "StorageOnly")]
internal OverwriteForce Force { get; init; } = OverwriteForce.None;

internal async Task ExecuteAsync(TextWriter writer)
ixje marked this conversation as resolved.
Show resolved Hide resolved
{
var (chainManager, _) = chainManagerFactory.LoadChain(Input);

if (chainManager.Chain.ConsensusNodes.Count != 1)
{
throw new ArgumentException("Contract download is only supported for single-node consensus");
}

var result = await NodeUtility.DownloadContractStateAsync(Contract, RpcUri, Height);
using var expressNode = chainManager.GetExpressNode();
await expressNode.PersistContractAsync(result.contractState, result.storagePairs, Force);
}

internal async Task<int> OnExecuteAsync(CommandLineApplication app, IConsole console)
{
devhawk marked this conversation as resolved.
Show resolved Hide resolved
try
{
await ExecuteAsync(console.Out).ConfigureAwait(false);
return 0;
}
catch (Exception ex)
{
app.WriteException(ex);
return 1;
}
}
}
}
}
2 changes: 1 addition & 1 deletion src/neoxp/Commands/ContractCommand.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
namespace NeoExpress.Commands
{
[Command("contract", Description = "Manage smart contracts")]
[Subcommand(typeof(Deploy), typeof(Get), typeof(Hash), typeof(Invoke), typeof(List), typeof(Run), typeof(Storage))]
[Subcommand(typeof(Deploy), typeof(Download), typeof(Get), typeof(Hash), typeof(Invoke), typeof(List), typeof(Run), typeof(Storage))]
partial class ContractCommand
{
internal int OnExecute(CommandLineApplication app, IConsole console)
Expand Down
44 changes: 44 additions & 0 deletions src/neoxp/Extensions/Extensions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
using McMaster.Extensions.CommandLineUtils;
using Neo;
using Neo.BlockchainToolkit;
using Neo.IO.Json;
using Neo.Network.P2P.Payloads;
using Neo.Network.RPC;
using Neo.Network.RPC.Models;
Expand Down Expand Up @@ -266,5 +267,48 @@ static bool TryGetNEP6Wallet(string path, string password, ProtocolSettings sett
}
}
}

// TODO: remove this copy of MakeFindStateParam https://github.com/neo-project/neo-express/issues/219
private static JObject[] MakeFindStatesParams(UInt256 rootHash, UInt160 scriptHash, ReadOnlySpan<byte> prefix,
ReadOnlySpan<byte> from = default, int? count = null)
{
var @params = new JObject[count.HasValue ? 5 : 4];
@params[0] = (JObject)rootHash.ToString();
@params[1] = (JObject)scriptHash.ToString();
@params[2] = (JObject)Convert.ToBase64String(prefix);
@params[3] = (JObject)Convert.ToBase64String(from);
if (count.HasValue)
@params[4] = (JObject)(double)count.Value;
return @params;
}

public static async Task<IReadOnlyList<(string key, string value)>> ExpressFindStatesAsync(this RpcClient rpcClient, UInt256 rootHash,
UInt160 contractScriptHash, ReadOnlyMemory<byte> prefix, ReadOnlyMemory<byte> from = default, int? pageSize = null)
{
var states = Enumerable.Empty<(string key, string value)>();
var start = from;

while (true)
{
var @params = MakeFindStatesParams(rootHash, contractScriptHash, prefix.Span, start.Span, pageSize);
var response = await rpcClient.RpcSendAsync("findstates", @params).ConfigureAwait(false);

var jsonResults = (JArray)response["results"];
if (jsonResults.Count == 0) break;

var results = jsonResults
.Select(j => (
j["key"].AsString(),
j["value"].AsString()
));
states = states.Concat(results);

var truncated = response["truncated"].AsBoolean();
if (truncated) break;
start = Convert.FromBase64String(jsonResults[jsonResults.Count - 1]["key"].AsString());
}

return states.ToList();
}
}
}
3 changes: 3 additions & 0 deletions src/neoxp/IExpressNode.cs
Original file line number Diff line number Diff line change
Expand Up @@ -6,10 +6,12 @@
using Neo.Cryptography.ECC;
using Neo.Network.P2P.Payloads;
using Neo.Network.RPC.Models;
using Neo.SmartContract;
using Neo.SmartContract.Manifest;
using Neo.SmartContract.Native;
using Neo.VM;
using Neo.Wallets;
using NeoExpress.Commands;
using NeoExpress.Models;

namespace NeoExpress
Expand Down Expand Up @@ -40,6 +42,7 @@ enum CheckpointMode { Online, Offline }
Task<IReadOnlyList<ExpressStorage>> ListStoragesAsync(UInt160 scriptHash);
Task<IReadOnlyList<TokenContract>> ListTokenContractsAsync();

Task<int> PersistContractAsync(ContractState state, IReadOnlyList<(string key, string value)> storagePairs, ContractCommand.OverwriteForce force);
IAsyncEnumerable<(uint blockIndex, NotificationRecord notification)> EnumerateNotificationsAsync(IReadOnlySet<UInt160>? contractFilter, IReadOnlySet<string>? eventFilter);
}
}
20 changes: 19 additions & 1 deletion src/neoxp/Node/Modules/ExpressRpcMethods.cs
Original file line number Diff line number Diff line change
Expand Up @@ -10,14 +10,18 @@
using Neo.IO;
using Neo.IO.Json;
using Neo.Network.P2P.Payloads;
using Neo.Network.RPC;
using Neo.Persistence;
using Neo.Plugins;
using Neo.SmartContract;
using Neo.SmartContract.Native;
using Neo.VM;
using Neo.Wallets;
using NeoExpress.Commands;
using NeoExpress.Models;
using ByteString = Neo.VM.Types.ByteString;
using RpcException = Neo.Plugins.RpcException;
using Utility = Neo.Utility;

namespace NeoExpress.Node
{
Expand Down Expand Up @@ -236,7 +240,7 @@ public JObject ExpressListTokenContracts(JArray _)
var height = NativeContract.Ledger.CurrentIndex(snapshot) + 1;
var oracleNodes = NativeContract.RoleManagement.GetDesignatedByRole(snapshot, Role.Oracle, height);
var request = NativeContract.Oracle.GetRequest(snapshot, response.Id);
var tx = ExpressOracle.CreateResponseTx(snapshot, request, response, oracleNodes, neoSystem.Settings);
var tx = NodeUtility.CreateResponseTx(snapshot, request, response, oracleNodes, neoSystem.Settings);
return tx == null ? null : Convert.ToBase64String(tx.ToArray());
}

Expand Down Expand Up @@ -521,6 +525,20 @@ public JObject GetNep11Properties(JArray @params)
}
return json;
}

[RpcMethod]
public JObject ExpressPersistContract(JObject @params)
{
var state = RpcClient.ContractStateFromJson(@params[0]["state"]);
var storagePairs = ((JArray)@params[0]["storage"])
.Select(s => (
s["key"].AsString(),
s["value"].AsString())
).ToArray();
var force = Enum.Parse<ContractCommand.OverwriteForce>(@params[0]["force"].AsString());

return NodeUtility.PersistContract(neoSystem.GetSnapshot(), state, storagePairs, force);
}

static readonly IReadOnlySet<string> nep11PropertyNames = new HashSet<string>
{
Expand Down
Loading