2022-09-19 17:00:46 +00:00
|
|
|
from typing import Optional
|
|
|
|
|
2023-01-10 13:02:24 +00:00
|
|
|
from frostfs_testlib.cli.cli_command import CliCommand
|
|
|
|
from frostfs_testlib.shell import CommandResult
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
|
2022-09-23 06:12:04 +00:00
|
|
|
class NeoGoContract(CliCommand):
|
2022-09-19 17:00:46 +00:00
|
|
|
def compile(
|
|
|
|
self,
|
|
|
|
input_file: str,
|
|
|
|
out: str,
|
|
|
|
manifest: str,
|
|
|
|
config: str,
|
|
|
|
no_standards: bool = False,
|
|
|
|
no_events: bool = False,
|
|
|
|
no_permissions: bool = False,
|
|
|
|
bindings: Optional[str] = None,
|
|
|
|
) -> CommandResult:
|
2022-10-05 09:14:51 +00:00
|
|
|
"""Compile a smart contract to a .nef file.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
input_file: Input file for the smart contract to be compiled.
|
|
|
|
out: Output of the compiled contract.
|
|
|
|
manifest: Emit contract manifest (*.manifest.json) file into separate file using
|
|
|
|
configuration input file (*.yml).
|
|
|
|
config: Configuration input file (*.yml).
|
|
|
|
no_standards: Do not check compliance with supported standards.
|
|
|
|
no_events: Do not check emitted events with the manifest.
|
|
|
|
no_permissions: Do not check if invoked contracts are allowed in manifest.
|
|
|
|
bindings: Output file for smart-contract bindings configuration.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
|
|
|
return self._execute(
|
|
|
|
"contract compile",
|
|
|
|
**{
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param not in ["self"]
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def deploy(
|
|
|
|
self,
|
|
|
|
address: str,
|
|
|
|
input_file: str,
|
|
|
|
manifest: str,
|
|
|
|
rpc_endpoint: str,
|
2022-10-26 09:40:57 +00:00
|
|
|
sysgas: Optional[float] = None,
|
2022-09-19 17:00:46 +00:00
|
|
|
wallet: Optional[str] = None,
|
|
|
|
wallet_config: Optional[str] = None,
|
2022-10-26 09:40:57 +00:00
|
|
|
wallet_password: Optional[str] = None,
|
2022-09-19 17:00:46 +00:00
|
|
|
gas: Optional[float] = None,
|
|
|
|
out: Optional[str] = None,
|
|
|
|
force: bool = False,
|
|
|
|
timeout: int = 10,
|
|
|
|
) -> CommandResult:
|
2022-09-23 06:12:04 +00:00
|
|
|
"""Deploy a smart contract (.nef with description)
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
wallet: Wallet to use to get the key for transaction signing;
|
|
|
|
conflicts with wallet_config.
|
|
|
|
wallet_config: Path to wallet config to use to get the key for transaction signing;
|
|
|
|
conflicts with wallet.
|
2022-10-26 09:40:57 +00:00
|
|
|
wallet_password: Wallet password.
|
2022-10-05 09:14:51 +00:00
|
|
|
address: Address to use as transaction signee (and gas source).
|
|
|
|
gas: Network fee to add to the transaction (prioritizing it).
|
|
|
|
sysgas: System fee to add to transaction (compensating for execution).
|
|
|
|
out: File to put JSON transaction to.
|
|
|
|
force: Do not ask for a confirmation.
|
|
|
|
rpc_endpoint: RPC node address.
|
|
|
|
timeout: Timeout for the operation (default: 10s).
|
|
|
|
input_file: Input file for the smart contract (*.nef).
|
|
|
|
manifest: Emit contract manifest (*.manifest.json) file into separate file using
|
|
|
|
configuration input file (*.yml).
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
|
|
|
assert bool(wallet) ^ bool(wallet_config), self.WALLET_SOURCE_ERROR_MSG
|
2022-10-26 09:40:57 +00:00
|
|
|
exec_param = {
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param not in ["self", "wallet_password"]
|
|
|
|
}
|
|
|
|
exec_param["timeout"] = f"{timeout}s"
|
|
|
|
|
|
|
|
if wallet_password is not None:
|
|
|
|
return self._execute_with_password(
|
|
|
|
"contract deploy",
|
|
|
|
wallet_password,
|
|
|
|
**exec_param,
|
|
|
|
)
|
|
|
|
if wallet_config:
|
|
|
|
return self._execute(
|
|
|
|
"contract deploy",
|
|
|
|
**exec_param,
|
|
|
|
)
|
|
|
|
|
|
|
|
raise Exception(self.WALLET_PASSWD_ERROR_MSG)
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
def generate_wrapper(
|
|
|
|
self,
|
|
|
|
out: str,
|
|
|
|
hash: str,
|
|
|
|
config: Optional[str] = None,
|
|
|
|
manifest: Optional[str] = None,
|
|
|
|
) -> CommandResult:
|
2022-10-05 09:14:51 +00:00
|
|
|
"""Generate wrapper to use in other contracts.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
config: Configuration file to use.
|
|
|
|
manifest: Read contract manifest (*.manifest.json) file.
|
|
|
|
out: Output of the compiled contract.
|
|
|
|
hash: Smart-contract hash.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
|
|
|
return self._execute(
|
|
|
|
"contract generate-wrapper",
|
|
|
|
**{
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param not in ["self"]
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def invokefunction(
|
|
|
|
self,
|
|
|
|
scripthash: str,
|
2022-10-26 09:40:57 +00:00
|
|
|
address: Optional[str] = None,
|
2022-09-19 17:00:46 +00:00
|
|
|
wallet: Optional[str] = None,
|
|
|
|
method: Optional[str] = None,
|
|
|
|
arguments: Optional[str] = None,
|
|
|
|
multisig_hash: Optional[str] = None,
|
|
|
|
wallet_config: Optional[str] = None,
|
2022-10-26 09:40:57 +00:00
|
|
|
wallet_password: Optional[str] = None,
|
2022-09-19 17:00:46 +00:00
|
|
|
gas: Optional[float] = None,
|
|
|
|
sysgas: Optional[float] = None,
|
|
|
|
out: Optional[str] = None,
|
|
|
|
force: bool = False,
|
|
|
|
rpc_endpoint: Optional[str] = None,
|
|
|
|
timeout: int = 10,
|
|
|
|
) -> CommandResult:
|
2022-10-05 09:14:51 +00:00
|
|
|
"""Executes given (as a script hash) deployed script.
|
|
|
|
|
|
|
|
Script is executed with the given method, arguments and signers. Sender is included in
|
|
|
|
the list of signers by default with None witness scope. If you'd like to change default
|
|
|
|
sender's scope, specify it via signers parameter. See testinvokefunction documentation
|
|
|
|
for the details about parameters. It differs from testinvokefunction in that this command
|
|
|
|
sends an invocation transaction to the network.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
scripthash: Function hash.
|
|
|
|
method: Call method.
|
|
|
|
arguments: Method arguments.
|
|
|
|
multisig_hash: Multisig hash.
|
|
|
|
wallet: Wallet to use to get the key for transaction signing;
|
|
|
|
conflicts with wallet_config.
|
|
|
|
wallet_config: Path to wallet config to use to get the key for transaction signing;
|
|
|
|
conflicts with wallet.
|
2022-10-26 09:40:57 +00:00
|
|
|
wallet_password: Wallet password.
|
2022-10-05 09:14:51 +00:00
|
|
|
address: Address to use as transaction signee (and gas source).
|
|
|
|
gas: Network fee to add to the transaction (prioritizing it).
|
|
|
|
sysgas: System fee to add to transaction (compensating for execution).
|
|
|
|
out: File to put JSON transaction to.
|
|
|
|
force: Force-push the transaction in case of bad VM state after test script invocation.
|
|
|
|
rpc_endpoint: RPC node address.
|
|
|
|
timeout: Timeout for the operation (default: 10s).
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
2022-10-26 09:40:57 +00:00
|
|
|
|
|
|
|
assert bool(wallet) ^ bool(wallet_config), self.WALLET_SOURCE_ERROR_MSG
|
|
|
|
|
2022-09-19 17:00:46 +00:00
|
|
|
multisig_hash = f"-- {multisig_hash}" or ""
|
2022-10-26 09:40:57 +00:00
|
|
|
post_data = f"{scripthash} {method or ''} {arguments or ''} {multisig_hash}"
|
|
|
|
exec_param = {
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param
|
|
|
|
not in [
|
|
|
|
"self",
|
|
|
|
"scripthash",
|
|
|
|
"method",
|
|
|
|
"arguments",
|
|
|
|
"multisig_hash",
|
|
|
|
"wallet_password",
|
|
|
|
]
|
|
|
|
}
|
|
|
|
exec_param["timeout"] = f"{timeout}s"
|
|
|
|
exec_param["post_data"] = post_data
|
|
|
|
if wallet_password is not None:
|
|
|
|
return self._execute_with_password(
|
|
|
|
"contract invokefunction", wallet_password, **exec_param
|
|
|
|
)
|
|
|
|
if wallet_config:
|
|
|
|
return self._execute("contract invokefunction", **exec_param)
|
|
|
|
|
|
|
|
raise Exception(self.WALLET_PASSWD_ERROR_MSG)
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
def testinvokefunction(
|
|
|
|
self,
|
|
|
|
scripthash: str,
|
|
|
|
wallet: Optional[str] = None,
|
2022-10-26 09:40:57 +00:00
|
|
|
wallet_password: Optional[str] = None,
|
2022-09-19 17:00:46 +00:00
|
|
|
method: Optional[str] = None,
|
|
|
|
arguments: Optional[str] = None,
|
|
|
|
multisig_hash: Optional[str] = None,
|
|
|
|
rpc_endpoint: Optional[str] = None,
|
|
|
|
timeout: int = 10,
|
|
|
|
) -> CommandResult:
|
2022-10-05 09:14:51 +00:00
|
|
|
"""Executes given (as a script hash) deployed script.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
2022-10-05 09:14:51 +00:00
|
|
|
Script is executed with the given method, arguments and signers (sender is not included
|
|
|
|
by default). If no method is given "" is passed to the script, if no arguments are given,
|
|
|
|
an empty array is passed, if no signers are given no array is passed. If signers are
|
|
|
|
specified, the first one of them is treated as a sender. All of the given arguments are
|
|
|
|
encapsulated into array before invoking the script. The script thus should follow the
|
|
|
|
regular convention of smart contract arguments (method string and an array of other
|
|
|
|
arguments).
|
|
|
|
See more information and samples in `neo-go contract testinvokefunction --help`.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
scripthash: Function hash.
|
2022-10-26 09:40:57 +00:00
|
|
|
wallet: Wallet to use for testinvoke.
|
|
|
|
wallet_password: Wallet password.
|
2022-10-05 09:14:51 +00:00
|
|
|
method: Call method.
|
|
|
|
arguments: Method arguments.
|
|
|
|
multisig_hash: Multisig hash.
|
|
|
|
rpc_endpoint: RPC node address.
|
|
|
|
timeout: Timeout for the operation (default: 10s).
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
2022-10-26 09:40:57 +00:00
|
|
|
multisig_hash = f"-- {multisig_hash}" if multisig_hash else ""
|
|
|
|
post_data = f"{scripthash} {method or ''} {arguments or ''} {multisig_hash}"
|
|
|
|
exec_param = {
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param
|
|
|
|
not in [
|
|
|
|
"self",
|
|
|
|
"scripthash",
|
|
|
|
"method",
|
|
|
|
"arguments",
|
|
|
|
"multisig_hash",
|
|
|
|
"wallet_password",
|
|
|
|
]
|
|
|
|
}
|
|
|
|
exec_param["timeout"] = f"{timeout}s"
|
|
|
|
exec_param["post_data"] = post_data
|
|
|
|
if wallet_password is not None:
|
|
|
|
return self._execute_with_password(
|
|
|
|
"contract testinvokefunction", wallet_password, **exec_param
|
|
|
|
)
|
|
|
|
|
|
|
|
return self._execute("contract testinvokefunction", **exec_param)
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
def testinvokescript(
|
|
|
|
self,
|
|
|
|
input_file: str,
|
|
|
|
rpc_endpoint: Optional[str] = None,
|
|
|
|
timeout: int = 10,
|
|
|
|
) -> CommandResult:
|
2022-10-05 09:14:51 +00:00
|
|
|
"""Executes given compiled AVM instructions in NEF format.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
2022-10-05 09:14:51 +00:00
|
|
|
Instructions are executed with the given set of signers not including sender by default.
|
|
|
|
See testinvokefunction documentation for the details about parameters.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
input_file: Input location of the .nef file that needs to be invoked.
|
|
|
|
rpc_endpoint: RPC node address.
|
|
|
|
timeout: Timeout for the operation (default: 10s).
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
2022-10-26 09:40:57 +00:00
|
|
|
exec_param = {
|
|
|
|
param: param_value for param, param_value in locals().items() if param not in ["self"]
|
|
|
|
}
|
|
|
|
exec_param["timeout"] = f"{timeout}s"
|
2022-09-19 17:00:46 +00:00
|
|
|
return self._execute(
|
2022-10-26 09:40:57 +00:00
|
|
|
"contract testinvokescript",
|
|
|
|
**exec_param,
|
2022-09-19 17:00:46 +00:00
|
|
|
)
|
|
|
|
|
2022-10-05 09:14:51 +00:00
|
|
|
def init(self, name: str, skip_details: bool = False) -> CommandResult:
|
|
|
|
"""Initialize a new smart-contract in a directory with boiler plate code.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
name: Name of the smart-contract to be initialized.
|
|
|
|
skip_details: Skip filling in the projects and contract details.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
|
|
|
return self._execute(
|
|
|
|
"contract init",
|
|
|
|
**{
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param not in ["self"]
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def inspect(
|
|
|
|
self,
|
|
|
|
input_file: Optional[str] = None,
|
|
|
|
compile: Optional[str] = None,
|
|
|
|
) -> CommandResult:
|
2022-10-05 09:14:51 +00:00
|
|
|
"""Creates a user readable dump of the program instructions.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
input_file: Input file of the program (either .go or .nef).
|
|
|
|
compile: Compile input file (it should be go code then).
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
|
|
|
return self._execute(
|
|
|
|
"contract inspect",
|
|
|
|
**{
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param not in ["self"]
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def calc_hash(
|
|
|
|
self,
|
|
|
|
input_file: str,
|
|
|
|
manifest: str,
|
|
|
|
sender: Optional[str] = None,
|
|
|
|
) -> CommandResult:
|
2022-10-05 09:14:51 +00:00
|
|
|
"""Calculates hash of a contract after deployment.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-05 09:14:51 +00:00
|
|
|
input_file: Path to NEF file.
|
|
|
|
sender: Sender script hash or address.
|
|
|
|
manifest: Path to manifest file.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
|
|
|
return self._execute(
|
|
|
|
"contract calc-hash",
|
|
|
|
**{
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param not in ["self"]
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def add_group(
|
|
|
|
self,
|
|
|
|
manifest: str,
|
|
|
|
address: str,
|
|
|
|
wallet: Optional[str] = None,
|
|
|
|
wallet_config: Optional[str] = None,
|
2022-10-26 09:40:57 +00:00
|
|
|
wallet_password: Optional[str] = None,
|
2022-09-19 17:00:46 +00:00
|
|
|
sender: Optional[str] = None,
|
|
|
|
nef: Optional[str] = None,
|
|
|
|
) -> CommandResult:
|
2022-10-05 09:14:51 +00:00
|
|
|
"""Adds group to the manifest.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Args:
|
2022-10-26 09:40:57 +00:00
|
|
|
wallet: Wallet to use to get the key for transaction signing;
|
|
|
|
conflicts with wallet_config.
|
|
|
|
wallet_config: Path to wallet config to use to get the key for transaction signing;
|
|
|
|
conflicts with wallet.
|
|
|
|
wallet_password: Wallet password.
|
2022-10-05 09:14:51 +00:00
|
|
|
sender: Deploy transaction sender.
|
|
|
|
address: Account to sign group with.
|
|
|
|
nef: Path to the NEF file.
|
|
|
|
manifest: Path to the manifest.
|
2022-09-19 17:00:46 +00:00
|
|
|
|
|
|
|
Returns:
|
2022-10-05 09:14:51 +00:00
|
|
|
Command's result.
|
2022-09-19 17:00:46 +00:00
|
|
|
"""
|
2022-10-26 09:40:57 +00:00
|
|
|
assert bool(wallet) ^ bool(wallet_config), self.WALLET_SOURCE_ERROR_MSG
|
|
|
|
exec_param = {
|
|
|
|
param: param_value
|
|
|
|
for param, param_value in locals().items()
|
|
|
|
if param not in ["self", "wallet_password"]
|
|
|
|
}
|
|
|
|
if wallet_password is not None:
|
|
|
|
return self._execute_with_password(
|
|
|
|
"contract manifest add-group", wallet_password, **exec_param
|
|
|
|
)
|
|
|
|
if wallet_config:
|
|
|
|
return self._execute("contract manifest add-group", **exec_param)
|
|
|
|
|
|
|
|
raise Exception(self.WALLET_PASSWD_ERROR_MSG)
|