From cbbde36cfc5e90b48f1b60815ebbf4c56b38ab5f Mon Sep 17 00:00:00 2001 From: Nando Vieira Date: Tue, 11 Jun 2024 11:25:07 -0700 Subject: [PATCH] Extract alias logic into its own implementation. --- .../src/commands/contract/deploy/wasm.rs | 57 +++--------- .../src/commands/contract/invoke.rs | 36 ++------ cmd/soroban-cli/src/commands/contract/mod.rs | 88 ++++++++++++++++++- 3 files changed, 106 insertions(+), 75 deletions(-) diff --git a/cmd/soroban-cli/src/commands/contract/deploy/wasm.rs b/cmd/soroban-cli/src/commands/contract/deploy/wasm.rs index ff56c0e50e..a5f58d437c 100644 --- a/cmd/soroban-cli/src/commands/contract/deploy/wasm.rs +++ b/cmd/soroban-cli/src/commands/contract/deploy/wasm.rs @@ -1,9 +1,6 @@ +use std::array::TryFromSliceError; use std::fmt::Debug; -use std::fs::{self, create_dir_all}; -use std::io::Write; use std::num::ParseIntError; -use std::path::PathBuf; -use std::{array::TryFromSliceError, fs::OpenOptions}; use clap::{arg, command, Parser}; use rand::Rng; @@ -18,7 +15,7 @@ use soroban_env_host::{ HostError, }; -use crate::commands::contract::AliasData; +use crate::commands::contract::{AliasData, AliasDataError}; use crate::commands::{ config::data, contract::{self, id::wasm::get_contract_id}, @@ -108,16 +105,12 @@ pub enum Error { Network(#[from] network::Error), #[error(transparent)] Wasm(#[from] wasm::Error), - #[error("cannot access config dir for alias file")] - CannotAccessConfigDir, #[error( "alias must be 1-30 chars long, and have only letters, numbers, underscores and dashes" )] InvalidAliasFormat { alias: String }, #[error(transparent)] - JsonSerialization(#[from] serde_json::Error), - #[error(transparent)] - Io(#[from] std::io::Error), + AliasData(#[from] AliasDataError), } impl Cmd { @@ -128,7 +121,13 @@ impl Cmd { match res { TxnEnvelopeResult::TxnEnvelope(tx) => println!("{}", tx.to_xdr_base64(Limits::none())?), TxnEnvelopeResult::Res(contract) => { - self.save_contract_id(&contract)?; + AliasData::save_contract_id( + &self.config.config_dir()?, + &contract, + self.alias.as_ref(), + &self.config.get_network()?.network_passphrase, + )?; + println!("{contract}"); } } @@ -149,42 +148,6 @@ impl Cmd { None => Ok(()), } } - - fn alias_path_for(&self, alias: &str) -> Result { - let config_dir = self.config.config_dir()?; - let file_name = format!("{alias}.json"); - - Ok(config_dir.join("contract-ids").join(file_name)) - } - - fn save_contract_id(&self, contract: &String) -> Result<(), Error> { - let Some(alias) = &self.alias else { - return Ok(()); - }; - - let file_path = self.alias_path_for(alias)?; - let dir = file_path.parent().ok_or(Error::CannotAccessConfigDir)?; - - create_dir_all(dir).map_err(|_| Error::CannotAccessConfigDir)?; - - let content = fs::read_to_string(&file_path).unwrap_or_default(); - let mut data: AliasData = serde_json::from_str(&content).unwrap_or_default(); - - let mut to_file = OpenOptions::new() - .create(true) - .truncate(true) - .write(true) - .open(file_path)?; - - data.ids.insert( - self.config.get_network()?.network_passphrase, - contract.into(), - ); - - let content = serde_json::to_string(&data)?; - - Ok(to_file.write_all(content.as_bytes())?) - } } #[async_trait::async_trait] diff --git a/cmd/soroban-cli/src/commands/contract/invoke.rs b/cmd/soroban-cli/src/commands/contract/invoke.rs index 712c73c511..2caaa7fcb9 100644 --- a/cmd/soroban-cli/src/commands/contract/invoke.rs +++ b/cmd/soroban-cli/src/commands/contract/invoke.rs @@ -31,7 +31,7 @@ use super::super::{ config::{self, locator}, events, }; -use super::AliasData; +use super::{AliasData, AliasDataError}; use crate::commands::txn_result::{TxnEnvelopeResult, TxnResult}; use crate::commands::NetworkRunnable; use crate::get_spec::{self, get_remote_contract_spec}; @@ -153,12 +153,8 @@ pub enum Error { Network(#[from] network::Error), #[error(transparent)] GetSpecError(#[from] get_spec::Error), - #[error("unable to read alias file")] - UnableToReadAliasFile, - #[error("alias file not found")] - NoAliasFileFound, #[error(transparent)] - JsonDeserialization(#[from] serde_json::Error), + AliasData(#[from] AliasDataError), } impl From for Error { @@ -321,28 +317,14 @@ impl Cmd { .map_err(|e| Error::CannotParseContractId(contract_id.clone(), e)) } - fn alias_path(&self) -> Result { - let config_dir = self.config.config_dir()?; - let file_name = format!("{}.json", self.contract_id); - - Ok(config_dir.join("contract-ids").join(file_name)) - } - fn load_contract_id(&self) -> Result, Error> { - let network = &self.config.get_network()?.network_passphrase; - let file_path = self.alias_path()?; - - if !file_path.exists() { - return Ok(None); - } - - let content = fs::read_to_string(file_path)?; - let data: AliasData = serde_json::from_str(&content)?; - - match data.ids.get(network) { - Some(id) => Ok(Some(id.into())), - _ => Ok(None), - } + let config_dir = self.config.config_dir()?; + let network_passphrase = &self.config.get_network()?.network_passphrase; + Ok(AliasData::get_contract_id( + &self.contract_id, + &config_dir, + network_passphrase, + )?) } } diff --git a/cmd/soroban-cli/src/commands/contract/mod.rs b/cmd/soroban-cli/src/commands/contract/mod.rs index 9da55714cb..7c0b23d30e 100644 --- a/cmd/soroban-cli/src/commands/contract/mod.rs +++ b/cmd/soroban-cli/src/commands/contract/mod.rs @@ -13,7 +13,12 @@ pub mod optimize; pub mod read; pub mod restore; -use std::collections::HashMap; +use std::{ + collections::HashMap, + fs::{self, create_dir_all, OpenOptions}, + io::Write, + path::{Path, PathBuf}, +}; use serde::{Deserialize, Serialize}; @@ -174,3 +179,84 @@ pub enum SpecOutput { pub struct AliasData { ids: HashMap, } + +#[derive(thiserror::Error, Debug)] +pub enum AliasDataError { + #[error("unable to read alias file")] + UnableToReadAliasFile, + #[error("alias file not found")] + NoAliasFileFound, + #[error(transparent)] + JsonDeserialization(#[from] serde_json::Error), + #[error(transparent)] + Io(#[from] std::io::Error), + #[error("cannot access config dir for alias file")] + CannotAccessConfigDir, +} + +impl AliasData { + pub fn load(config_dir: &Path, alias: Option<&str>) -> Result, AliasDataError> { + let Some(alias) = alias else { + return Ok(None); + }; + + let path = Self::alias_path(config_dir, alias); + let content = fs::read_to_string(path)?; + let data: Self = serde_json::from_str(&content).unwrap_or_default(); + + Ok(Some(data)) + } + + pub fn alias_path(config_dir: &Path, alias: &str) -> PathBuf { + let file_name = format!("{alias}.json"); + config_dir.join("contract-ids").join(file_name) + } + + pub fn save_contract_id( + config_dir: &Path, + contract_id: &str, + alias: Option<&String>, + network_passphrase: &str, + ) -> Result<(), AliasDataError> { + let Some(alias) = alias else { + return Ok(()); + }; + + let path = Self::alias_path(config_dir, alias); + let dir = path.parent().ok_or(AliasDataError::CannotAccessConfigDir)?; + + create_dir_all(dir).map_err(|_| AliasDataError::CannotAccessConfigDir)?; + + let content = fs::read_to_string(&path).unwrap_or_default(); + let mut data: Self = serde_json::from_str(&content).unwrap_or_default(); + + let mut to_file = OpenOptions::new() + .create(true) + .truncate(true) + .write(true) + .open(path)?; + + data.ids + .insert(network_passphrase.into(), contract_id.into()); + + let content = serde_json::to_string(&data)?; + + Ok(to_file.write_all(content.as_bytes())?) + } + + pub fn get_contract_id( + alias: &str, + config_dir: &Path, + network_passphrase: &str, + ) -> Result, AliasDataError> { + let alias_data = Self::load(config_dir, Some(alias))?; + let Some(alias_data) = alias_data else { + return Ok(None); + }; + + match alias_data.ids.get(network_passphrase) { + Some(id) => Ok(Some(id.into())), + _ => Ok(None), + } + } +}