identity_iota_interaction/
keytool_signer.rsuse std::path::Path;
use std::path::PathBuf;
use crate::types::base_types::IotaAddress;
use crate::types::crypto::PublicKey;
use crate::types::crypto::Signature;
use crate::types::crypto::SignatureScheme;
use crate::IotaKeySignature;
use crate::TransactionDataBcs;
use anyhow::anyhow;
use anyhow::Context as _;
use async_trait::async_trait;
use fastcrypto::encoding::Base64;
use fastcrypto::encoding::Encoding;
use jsonpath_rust::JsonPathQuery as _;
use secret_storage::Error as SecretStorageError;
use secret_storage::Signer;
use serde::Deserialize;
use serde_json::Value;
#[derive(Debug, Default)]
pub struct KeytoolSignerBuilder {
address: Option<IotaAddress>,
iota_bin: Option<PathBuf>,
}
impl KeytoolSignerBuilder {
pub fn new() -> Self {
Self::default()
}
pub fn with_address(mut self, address: IotaAddress) -> Self {
self.address = Some(address);
self
}
pub fn iota_bin_location(mut self, path: impl AsRef<Path>) -> Self {
let path = path.as_ref().to_path_buf();
self.iota_bin = Some(path);
self
}
pub async fn build(self) -> anyhow::Result<KeytoolSigner> {
let KeytoolSignerBuilder { address, iota_bin } = self;
let iota_bin = iota_bin.unwrap_or_else(|| "iota".into());
let address = if let Some(address) = address {
address
} else {
get_active_address(&iota_bin).await?
};
let public_key = get_key(&iota_bin, address).await.context("cannot find key")?;
Ok(KeytoolSigner {
public_key,
iota_bin,
address,
})
}
}
#[derive(Debug)]
pub struct KeytoolSigner {
public_key: PublicKey,
iota_bin: PathBuf,
address: IotaAddress,
}
impl KeytoolSigner {
pub fn builder() -> KeytoolSignerBuilder {
KeytoolSignerBuilder::default()
}
pub fn address(&self) -> IotaAddress {
self.address
}
pub fn public_key(&self) -> &PublicKey {
&self.public_key
}
async fn run_iota_cli_command(&self, args: &str) -> anyhow::Result<Value> {
run_iota_cli_command_with_bin(&self.iota_bin, args).await
}
}
#[cfg_attr(feature = "send-sync-transaction", async_trait)]
#[cfg_attr(not(feature = "send-sync-transaction"), async_trait(?Send))]
impl Signer<IotaKeySignature> for KeytoolSigner {
type KeyId = IotaAddress;
fn key_id(&self) -> &Self::KeyId {
&self.address
}
async fn public_key(&self) -> Result<PublicKey, SecretStorageError> {
Ok(self.public_key.clone())
}
async fn sign(&self, data: &TransactionDataBcs) -> Result<Signature, SecretStorageError> {
let base64_data = Base64::encode(data);
let command = format!("keytool sign --address {} --data {base64_data}", self.address);
self
.run_iota_cli_command(&command)
.await
.and_then(|json| {
json
.get("iotaSignature")
.context("invalid JSON output: missing iotaSignature")?
.as_str()
.context("not a JSON string")?
.parse()
.map_err(|e| anyhow!("invalid IOTA signature: {e}"))
})
.map_err(SecretStorageError::Other)
}
}
async fn run_iota_cli_command_with_bin(iota_bin: impl AsRef<Path>, args: &str) -> anyhow::Result<Value> {
let iota_bin = iota_bin.as_ref();
cfg_if::cfg_if! {
if #[cfg(not(target_arch = "wasm32"))] {
let output = tokio::process::Command::new(iota_bin)
.args(args.split_ascii_whitespace())
.arg("--json")
.output()
.await
.map_err(|e| anyhow!("failed to run command: {e}"))?;
if !output.status.success() {
let err_msg =
String::from_utf8(output.stderr).map_err(|e| anyhow!("command failed with non-utf8 error message: {e}"))?;
return Err(anyhow!("failed to run \"iota client active-address\": {err_msg}"));
}
serde_json::from_slice(&output.stdout).context("invalid JSON object in command output")
} else {
extern "Rust" {
fn __wasm_exec_iota_cmd(cmd: &str) -> anyhow::Result<Value>;
}
let iota_bin = iota_bin.to_str().context("invalid IOTA bin path")?;
let cmd = format!("{iota_bin} {args} --json");
unsafe { __wasm_exec_iota_cmd(&cmd) }
}
}
}
async fn get_active_address(iota_bin: impl AsRef<Path>) -> anyhow::Result<IotaAddress> {
run_iota_cli_command_with_bin(iota_bin, "client active-address")
.await
.and_then(|value| serde_json::from_value(value).context("failed to parse IotaAddress from output"))
}
async fn get_key(iota_bin: impl AsRef<Path>, address: IotaAddress) -> anyhow::Result<PublicKey> {
let query = format!("$[?(@.iotaAddress==\"{}\")]", address);
let pk_json_data = run_iota_cli_command_with_bin(iota_bin, "keytool list")
.await
.and_then(|json_value| {
json_value
.path(&query)
.map_err(|e| anyhow!("failed to query JSON output: {e}"))?
.get_mut(0)
.context("key not found")
.map(Value::take)
})?;
let Ok(KeytoolPublicKeyHelper {
public_base64_key,
flag,
}) = serde_json::from_value(pk_json_data)
else {
return Err(anyhow!("invalid key format"));
};
let signature_scheme = SignatureScheme::from_flag_byte(&flag).context(format!("invalid signature flag `{flag}`"))?;
let pk_bytes = Base64::decode(&public_base64_key).context("invalid base64 encoding for key")?;
PublicKey::try_from_bytes(signature_scheme, &pk_bytes).map_err(|e| anyhow!("{e:?}"))
}
#[derive(Deserialize)]
#[serde(rename_all = "camelCase")]
struct KeytoolPublicKeyHelper {
public_base64_key: String,
flag: u8,
}