# Copyright © 2022, 2023 Cerc # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Affero General Public License for more details. # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see . from app.util import get_yaml from app.deploy_types import DeployCommandContext, LaconicStackSetupCommand from app.stack_state import State from app.deploy_util import VolumeMapping, run_container_command from app.command_types import CommandOptions from enum import Enum from pathlib import Path from shutil import copyfile, copytree import json import os import sys import tomli default_spec_file_content = """config: node_moniker: my-node-name chain_id: my-chain-id """ init_help_text = """Add helpful text here on setting config variables. """ class SetupPhase(Enum): INITIALIZE = 1 JOIN = 2 CREATE = 3 ILLEGAL = 3 def _config_toml_path(network_dir: Path): return network_dir.joinpath("config","client.toml") def _get_chain_id_from_config(network_dir: Path): chain_id = None with open(_config_toml_path(network_dir), "rb") as f: toml_dict = tomli.load(f) chain_id = toml_dict["chain-id"] return chain_id def _get_node_moniker_from_config(network_dir: Path): moniker = None with open(_config_toml_path(network_dir), "rb") as f: toml_dict = tomli.load(f) moniker = toml_dict["moniker"] return moniker def _get_node_key_from_gentx(options: CommandOptions, gentx_file_name: str): gentx_file_path = Path(gentx_file_name) if gentx_file_path.exists(): with open(Path(gentx_file_name), "rb") as f: parsed_json = json.load(f) return parsed_json['body']['messages'][0]['delegator_address'] else: print(f"Error: gentx file: {gentx_file_name} does not exist") sys.exit(1) def _comma_delimited_to_list(list_str: str): return list_str.split(",") if list_str else [] def _get_node_keys_from_gentx_files(options: CommandOptions, gentx_file_list: str): node_keys = [] gentx_files = _comma_delimited_to_list(gentx_file_list) for gentx_file in gentx_files: node_key = _get_node_key_from_gentx(options, gentx_file) if node_key: node_keys.append(node_key) return node_keys def _copy_gentx_files(options: CommandOptions, network_dir: Path, gentx_file_list: str): gentx_files = _comma_delimited_to_list(gentx_file_list) for gentx_file in gentx_files: gentx_file_path = Path(gentx_file) copyfile(gentx_file_path, os.path.join(network_dir, "config", "gentx", os.path.basename(gentx_file_path))) def setup(command_context: DeployCommandContext, parameters: LaconicStackSetupCommand, extra_args): options = command_context.cluster_context.options if options.debug: print(f"parameters: {parameters}") phase = SetupPhase.ILLEGAL if parameters.initialize_network: if parameters.join_network or parameters.create_network: print("Can't supply --join-network or --create-network with --initialize-network") sys.exit(1) if not parameters.chain_id: print("--chain-id is required") sys.exit(1) # node_moniker must be supplied if not parameters.node_moniker: print("Error: --node-moniker is required") sys.exit(1) phase = SetupPhase.INITIALIZE elif parameters.join_network: if parameters.initialize_network or parameters.create_network: print("Can't supply --initialize-network or --create-network with --join-network") sys.exit(1) phase = SetupPhase.JOIN elif parameters.create_network: if parameters.initialize_network or parameters.join_network: print("Can't supply --initialize-network or --join-network with --create-network") sys.exit(1) phase = SetupPhase.CREATE network_dir = Path(parameters.network_dir).absolute() laconicd_home_path_in_container = "/laconicd-home" mounts = [ VolumeMapping(network_dir, laconicd_home_path_in_container) ] if phase == SetupPhase.INITIALIZE: # We want to create the directory so if it exists that's an error if os.path.exists(network_dir): print(f"Error: network directory {network_dir} already exists") sys.exit(1) os.mkdir(network_dir) output, status = run_container_command( command_context, "laconicd", f"laconicd init {parameters.node_moniker} --home {laconicd_home_path_in_container}\ --chain-id {parameters.chain_id}", mounts) if options.debug: print(f"Command output: {output}") elif phase == SetupPhase.JOIN: if not os.path.exists(network_dir): print(f"Error: network directory {network_dir} doesn't exist") sys.exit(1) # Get the chain_id from the config file created in the INITIALIZE phase chain_id = _get_chain_id_from_config(network_dir) output1, status1 = run_container_command( command_context, "laconicd", f"laconicd keys add {parameters.key_name} --home {laconicd_home_path_in_container}\ --keyring-backend test", mounts) if options.debug: print(f"Command output: {output1}") output2, status2 = run_container_command( command_context, "laconicd", f"laconicd add-genesis-account {parameters.key_name} 12900000000000000000000achk\ --home {laconicd_home_path_in_container} --keyring-backend test", mounts) print(f"Command output: {output2}") output3, status3 = run_container_command( command_context, "laconicd", f"laconicd gentx {parameters.key_name} 90000000000achk --home {laconicd_home_path_in_container}\ --chain-id {chain_id} --keyring-backend test", mounts) print(f"Command output: {output3}") output4, status4 = run_container_command( command_context, "laconicd", f"laconicd keys show {parameters.key_name} -a --home {laconicd_home_path_in_container} --keyring-backend test", mounts) print(f"Command output: {output4}") elif phase == SetupPhase.CREATE: if not os.path.exists(network_dir): print(f"Error: network directory {network_dir} doesn't exist") sys.exit(1) # In the CREATE phase, we are either a "coordinator" node, generating the genesis.json file ourselves # OR we are a "not-coordinator" node, consuming a genesis file we got from the coordinator node. if parameters.genesis_file: # We got the genesis file from elsewhere # Copy it into our network dir genesis_file_path = Path(parameters.genesis_file) if not os.path.exists(genesis_file_path): print(f"Error: supplied genesis file: {parameters.genesis_file} does not exist.") sys.exit(1) copyfile(genesis_file_path, os.path.join(network_dir, "config", os.path.basename(genesis_file_path))) else: # We're generating the genesis file # First look in the supplied gentx files for the other nodes' keys other_node_keys = _get_node_keys_from_gentx_files(options, parameters.gentx_file_list) # Add those keys to our genesis, with balances we determine here (why?) for other_node_key in other_node_keys: outputk, statusk = run_container_command( command_context, "laconicd", f"laconicd add-genesis-account {other_node_key} 12900000000000000000000achk\ --home {laconicd_home_path_in_container} --keyring-backend test", mounts) print(f"Command output: {outputk}") # Copy the gentx json files into our network dir _copy_gentx_files(options, network_dir, parameters.gentx_file_list) # Now we can run collect-gentxs output1, status1 = run_container_command( command_context, "laconicd", f"laconicd collect-gentxs --home {laconicd_home_path_in_container}", mounts) print(f"Command output: {output1}") print(f"Generated genesis file, please copy to other nodes as required: \ {os.path.join(network_dir, 'config', 'genesis.json')}") # In both cases we validate the genesis file now output2, status1 = run_container_command( command_context, "laconicd", f"laconicd validate-genesis --home {laconicd_home_path_in_container}", mounts) print(f"Command output: {output2}") else: print("Illegal parameters supplied") sys.exit(1) def create(command_context: DeployCommandContext, extra_args): network_dir = extra_args if network_dir is None: print("Error: --network-dir must be supplied") sys.exit(1) network_dir_path = Path(network_dir) if not (network_dir_path.exists() and network_dir_path.is_dir()): print(f"Error: supplied network directory does not exist: {network_dir}") sys.exit(1) # Copy the network directory contents into our deployment # TODO: change this to work with non local paths deployment_config_dir = command_context.deployment_dir.joinpath("data", "laconicd-config") copytree(network_dir_path, deployment_config_dir, dirs_exist_ok=True) def init(command_context: DeployCommandContext): print(init_help_text) yaml = get_yaml() return yaml.load(default_spec_file_content) def get_state(command_context: DeployCommandContext): print("Here we get state") return State.CONFIGURED def change_state(command_context: DeployCommandContext): pass