mirror of
https://gitlab.com/pulsechaincom/staking-deposit-cli.git
synced 2025-01-12 05:50:06 +00:00
203 lines
7.2 KiB
Python
203 lines
7.2 KiB
Python
import os
|
|
import click
|
|
import json
|
|
from typing import (
|
|
Any,
|
|
Sequence,
|
|
)
|
|
|
|
from eth_typing import HexAddress
|
|
|
|
from staking_deposit.credentials import (
|
|
CredentialList,
|
|
)
|
|
from staking_deposit.utils.validation import (
|
|
validate_bls_withdrawal_credentials_list,
|
|
validate_bls_withdrawal_credentials_matching,
|
|
validate_eth1_withdrawal_address,
|
|
validate_int_range,
|
|
verify_bls_to_execution_change_json,
|
|
validate_validator_indices,
|
|
)
|
|
from staking_deposit.utils.constants import (
|
|
DEFAULT_BLS_TO_EXECUTION_CHANGES_FOLDER_NAME,
|
|
MAX_DEPOSIT_AMOUNT,
|
|
)
|
|
from staking_deposit.utils.click import (
|
|
captive_prompt_callback,
|
|
choice_prompt_func,
|
|
jit_option,
|
|
)
|
|
from staking_deposit.exceptions import ValidationError
|
|
from staking_deposit.utils.intl import (
|
|
closest_match,
|
|
load_text,
|
|
)
|
|
from staking_deposit.settings import (
|
|
ALL_CHAINS,
|
|
MAINNET,
|
|
PRATER,
|
|
get_chain_setting,
|
|
get_devnet_chain_setting,
|
|
)
|
|
from .existing_mnemonic import (
|
|
load_mnemonic_arguments_decorator,
|
|
)
|
|
|
|
|
|
def get_password(text: str) -> str:
|
|
return click.prompt(text, hide_input=True, show_default=False, type=str)
|
|
|
|
|
|
FUNC_NAME = 'generate_bls_to_execution_change'
|
|
|
|
|
|
@click.command(
|
|
help=load_text(['arg_generate_bls_to_execution_change', 'help'], func=FUNC_NAME),
|
|
)
|
|
@jit_option(
|
|
default=os.getcwd(),
|
|
help=lambda: load_text(['arg_bls_to_execution_changes_folder', 'help'], func=FUNC_NAME),
|
|
param_decls='--bls_to_execution_changes_folder',
|
|
type=click.Path(exists=True, file_okay=False, dir_okay=True),
|
|
)
|
|
@jit_option(
|
|
callback=captive_prompt_callback(
|
|
lambda x: closest_match(x, list(ALL_CHAINS.keys())),
|
|
choice_prompt_func(
|
|
lambda: load_text(['arg_chain', 'prompt'], func=FUNC_NAME),
|
|
list(ALL_CHAINS.keys())
|
|
),
|
|
),
|
|
default=MAINNET,
|
|
help=lambda: load_text(['arg_chain', 'help'], func=FUNC_NAME),
|
|
param_decls='--chain',
|
|
prompt=choice_prompt_func(
|
|
lambda: load_text(['arg_chain', 'prompt'], func=FUNC_NAME),
|
|
# Since `prater` is alias of `goerli`, do not show `prater` in the prompt message.
|
|
list(key for key in ALL_CHAINS.keys() if key != PRATER)
|
|
),
|
|
)
|
|
@load_mnemonic_arguments_decorator
|
|
@jit_option(
|
|
callback=captive_prompt_callback(
|
|
lambda num: validate_int_range(num, 0, 2**32),
|
|
lambda: load_text(['arg_validator_start_index', 'prompt'], func=FUNC_NAME),
|
|
),
|
|
default=0,
|
|
help=lambda: load_text(['arg_validator_start_index', 'help'], func=FUNC_NAME),
|
|
param_decls="--validator_start_index",
|
|
prompt=lambda: load_text(['arg_validator_start_index', 'prompt'], func=FUNC_NAME),
|
|
)
|
|
@jit_option(
|
|
callback=captive_prompt_callback(
|
|
lambda validator_indices: validate_validator_indices(validator_indices),
|
|
lambda: load_text(['arg_validator_indices', 'prompt'], func=FUNC_NAME),
|
|
),
|
|
help=lambda: load_text(['arg_validator_indices', 'help'], func=FUNC_NAME),
|
|
param_decls='--validator_indices',
|
|
prompt=lambda: load_text(['arg_validator_indices', 'prompt'], func=FUNC_NAME),
|
|
)
|
|
@jit_option(
|
|
callback=captive_prompt_callback(
|
|
lambda bls_withdrawal_credentials_list:
|
|
validate_bls_withdrawal_credentials_list(bls_withdrawal_credentials_list),
|
|
lambda: load_text(['arg_bls_withdrawal_credentials_list', 'prompt'], func=FUNC_NAME),
|
|
),
|
|
help=lambda: load_text(['arg_bls_withdrawal_credentials_list', 'help'], func=FUNC_NAME),
|
|
param_decls='--bls_withdrawal_credentials_list',
|
|
prompt=lambda: load_text(['arg_bls_withdrawal_credentials_list', 'prompt'], func=FUNC_NAME),
|
|
)
|
|
@jit_option(
|
|
callback=captive_prompt_callback(
|
|
lambda address: validate_eth1_withdrawal_address(None, None, address),
|
|
lambda: load_text(['arg_execution_address', 'prompt'], func=FUNC_NAME),
|
|
lambda: load_text(['arg_execution_address', 'confirm'], func=FUNC_NAME),
|
|
lambda: load_text(['arg_execution_address', 'mismatch'], func=FUNC_NAME),
|
|
),
|
|
help=lambda: load_text(['arg_execution_address', 'help'], func=FUNC_NAME),
|
|
param_decls='--execution_address',
|
|
prompt=lambda: load_text(['arg_execution_address', 'prompt'], func=FUNC_NAME),
|
|
)
|
|
@jit_option(
|
|
# Only for devnet tests
|
|
default=None,
|
|
help="[DEVNET ONLY] Set specific GENESIS_FORK_VERSION value",
|
|
param_decls='--devnet_chain_setting',
|
|
)
|
|
@click.pass_context
|
|
def generate_bls_to_execution_change(
|
|
ctx: click.Context,
|
|
bls_to_execution_changes_folder: str,
|
|
chain: str,
|
|
mnemonic: str,
|
|
mnemonic_password: str,
|
|
validator_start_index: int,
|
|
validator_indices: Sequence[int],
|
|
bls_withdrawal_credentials_list: Sequence[bytes],
|
|
execution_address: HexAddress,
|
|
devnet_chain_setting: str,
|
|
**kwargs: Any) -> None:
|
|
# Generate folder
|
|
bls_to_execution_changes_folder = os.path.join(
|
|
bls_to_execution_changes_folder,
|
|
DEFAULT_BLS_TO_EXECUTION_CHANGES_FOLDER_NAME,
|
|
)
|
|
if not os.path.exists(bls_to_execution_changes_folder):
|
|
os.mkdir(bls_to_execution_changes_folder)
|
|
|
|
# Get chain setting
|
|
chain_setting = get_chain_setting(chain)
|
|
|
|
if devnet_chain_setting is not None:
|
|
click.echo('\n%s\n' % '**[Warning] Using devnet chain setting to generate the SignedBLSToExecutionChange.**\t')
|
|
devnet_chain_setting_dict = json.loads(devnet_chain_setting)
|
|
chain_setting = get_devnet_chain_setting(
|
|
network_name=devnet_chain_setting_dict['network_name'],
|
|
genesis_fork_version=devnet_chain_setting_dict['genesis_fork_version'],
|
|
genesis_validator_root=devnet_chain_setting_dict['genesis_validator_root'],
|
|
)
|
|
|
|
if len(validator_indices) != len(bls_withdrawal_credentials_list):
|
|
raise ValueError(
|
|
"The size of `validator_indices` (%d) should be as same as `bls_withdrawal_credentials_list` (%d)."
|
|
% (len(validator_indices), len(bls_withdrawal_credentials_list))
|
|
)
|
|
|
|
num_validators = len(validator_indices)
|
|
amounts = [MAX_DEPOSIT_AMOUNT] * num_validators
|
|
|
|
credentials = CredentialList.from_mnemonic(
|
|
mnemonic=mnemonic,
|
|
mnemonic_password=mnemonic_password,
|
|
num_keys=num_validators,
|
|
amounts=amounts,
|
|
chain_setting=chain_setting,
|
|
start_index=validator_start_index,
|
|
hex_eth1_withdrawal_address=execution_address,
|
|
)
|
|
|
|
# Check if the given old bls_withdrawal_credentials is as same as the mnemonic generated
|
|
for i, credential in enumerate(credentials.credentials):
|
|
try:
|
|
validate_bls_withdrawal_credentials_matching(bls_withdrawal_credentials_list[i], credential)
|
|
except ValidationError as e:
|
|
click.echo('\n[Error] ' + str(e))
|
|
return
|
|
|
|
btec_file = credentials.export_bls_to_execution_change_json(bls_to_execution_changes_folder, validator_indices)
|
|
|
|
json_file_validation_result = verify_bls_to_execution_change_json(
|
|
btec_file,
|
|
credentials.credentials,
|
|
input_validator_indices=validator_indices,
|
|
input_execution_address=execution_address,
|
|
chain_setting=chain_setting,
|
|
)
|
|
if not json_file_validation_result:
|
|
raise ValidationError(load_text(['err_verify_btec']))
|
|
|
|
click.echo(load_text(['msg_creation_success']) + str(bls_to_execution_changes_folder))
|
|
|
|
click.pause(load_text(['msg_pause']))
|