alloy_sol_types/types/function.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138
use crate::{
abi::{Token, TokenSeq},
private::SolTypeValue,
Result, SolType, Word,
};
use alloc::vec::Vec;
/// A Solidity function call.
///
/// # Implementer's Guide
///
/// It should not be necessary to implement this trait manually. Instead, use
/// the [`sol!`](crate::sol!) procedural macro to parse Solidity syntax into
/// types that implement this trait.
pub trait SolCall: Sized {
/// The underlying tuple type which represents this type's arguments.
///
/// If this type has no arguments, this will be the unit type `()`.
type Parameters<'a>: SolType<Token<'a> = Self::Token<'a>>;
/// The arguments' corresponding [TokenSeq] type.
type Token<'a>: TokenSeq<'a>;
/// The function's return struct.
type Return;
/// The underlying tuple type which represents this type's return values.
///
/// If this type has no return values, this will be the unit type `()`.
type ReturnTuple<'a>: SolType<Token<'a> = Self::ReturnToken<'a>>;
/// The returns' corresponding [TokenSeq] type.
type ReturnToken<'a>: TokenSeq<'a>;
/// The function's ABI signature.
const SIGNATURE: &'static str;
/// The function selector: `keccak256(SIGNATURE)[0..4]`
const SELECTOR: [u8; 4];
/// Convert from the tuple type used for ABI encoding and decoding.
fn new(tuple: <Self::Parameters<'_> as SolType>::RustType) -> Self;
/// Tokenize the call's arguments.
fn tokenize(&self) -> Self::Token<'_>;
/// The size of the encoded data in bytes, **without** its selector.
#[inline]
fn abi_encoded_size(&self) -> usize {
if let Some(size) = <Self::Parameters<'_> as SolType>::ENCODED_SIZE {
return size;
}
// `total_words` includes the first dynamic offset which we ignore.
let offset = <<Self::Parameters<'_> as SolType>::Token<'_> as Token>::DYNAMIC as usize * 32;
(self.tokenize().total_words() * Word::len_bytes()).saturating_sub(offset)
}
/// ABI decode this call's arguments from the given slice, **without** its
/// selector.
#[inline]
fn abi_decode_raw(data: &[u8], validate: bool) -> Result<Self> {
<Self::Parameters<'_> as SolType>::abi_decode_sequence(data, validate).map(Self::new)
}
/// ABI decode this call's arguments from the given slice, **with** the
/// selector.
#[inline]
fn abi_decode(data: &[u8], validate: bool) -> Result<Self> {
let data = data
.strip_prefix(&Self::SELECTOR)
.ok_or_else(|| crate::Error::type_check_fail_sig(data, Self::SIGNATURE))?;
Self::abi_decode_raw(data, validate)
}
/// ABI encode the call to the given buffer **without** its selector.
#[inline]
fn abi_encode_raw(&self, out: &mut Vec<u8>) {
out.reserve(self.abi_encoded_size());
out.extend(crate::abi::encode_sequence(&self.tokenize()));
}
/// ABI encode the call to the given buffer **with** its selector.
#[inline]
fn abi_encode(&self) -> Vec<u8> {
let mut out = Vec::with_capacity(4 + self.abi_encoded_size());
out.extend(&Self::SELECTOR);
self.abi_encode_raw(&mut out);
out
}
/// ABI decode this call's return values from the given slice.
fn abi_decode_returns(data: &[u8], validate: bool) -> Result<Self::Return>;
/// ABI encode the call's return values.
#[inline]
fn abi_encode_returns<'a, E>(e: &'a E) -> Vec<u8>
where
E: SolTypeValue<Self::ReturnTuple<'a>>,
{
crate::abi::encode_sequence(&e.stv_to_tokens())
}
}
/// A Solidity constructor.
pub trait SolConstructor: Sized {
/// The underlying tuple type which represents this type's arguments.
///
/// If this type has no arguments, this will be the unit type `()`.
type Parameters<'a>: SolType<Token<'a> = Self::Token<'a>>;
/// The arguments' corresponding [TokenSeq] type.
type Token<'a>: TokenSeq<'a>;
/// Convert from the tuple type used for ABI encoding and decoding.
fn new(tuple: <Self::Parameters<'_> as SolType>::RustType) -> Self;
/// Tokenize the call's arguments.
fn tokenize(&self) -> Self::Token<'_>;
/// The size of the encoded data in bytes.
#[inline]
fn abi_encoded_size(&self) -> usize {
if let Some(size) = <Self::Parameters<'_> as SolType>::ENCODED_SIZE {
return size;
}
// `total_words` includes the first dynamic offset which we ignore.
let offset = <<Self::Parameters<'_> as SolType>::Token<'_> as Token>::DYNAMIC as usize * 32;
(self.tokenize().total_words() * Word::len_bytes()).saturating_sub(offset)
}
/// ABI encode the call to the given buffer.
#[inline]
fn abi_encode(&self) -> Vec<u8> {
crate::abi::encode_sequence(&self.tokenize())
}
}