use crate::feed_id::ID;
use crate::report::base::{ReportBase, ReportError};
use num_bigint::BigInt;
#[derive(Debug)]
pub struct ReportDataV8 {
pub feed_id: ID,
pub valid_from_timestamp: u32,
pub observations_timestamp: u32,
pub native_fee: BigInt,
pub link_fee: BigInt,
pub expires_at: u32,
pub last_update_timestamp: u64,
pub mid_price: BigInt,
pub market_status: u32,
}
impl ReportDataV8 {
pub fn decode(data: &[u8]) -> Result<Self, ReportError> {
if data.len() < 9 * ReportBase::WORD_SIZE {
return Err(ReportError::DataTooShort("ReportDataV8"));
}
let feed_id = ID(data[..ReportBase::WORD_SIZE]
.try_into()
.map_err(|_| ReportError::InvalidLength("feed_id (bytes32)"))?);
let valid_from_timestamp = ReportBase::read_uint32(data, ReportBase::WORD_SIZE)?;
let observations_timestamp = ReportBase::read_uint32(data, 2 * ReportBase::WORD_SIZE)?;
let native_fee = ReportBase::read_uint192(data, 3 * ReportBase::WORD_SIZE)?;
let link_fee = ReportBase::read_uint192(data, 4 * ReportBase::WORD_SIZE)?;
let expires_at = ReportBase::read_uint32(data, 5 * ReportBase::WORD_SIZE)?;
let last_update_timestamp = ReportBase::read_uint64(data, 6 * ReportBase::WORD_SIZE)?;
let mid_price = ReportBase::read_int192(data, 7 * ReportBase::WORD_SIZE)?;
let market_status = ReportBase::read_uint32(data, 8 * ReportBase::WORD_SIZE)?;
Ok(Self {
feed_id,
valid_from_timestamp,
observations_timestamp,
native_fee,
link_fee,
expires_at,
last_update_timestamp,
mid_price,
market_status,
})
}
pub fn abi_encode(&self) -> Result<Vec<u8>, ReportError> {
let mut buffer = Vec::with_capacity(9 * ReportBase::WORD_SIZE);
buffer.extend_from_slice(&self.feed_id.0);
buffer.extend_from_slice(&ReportBase::encode_uint32(self.valid_from_timestamp)?);
buffer.extend_from_slice(&ReportBase::encode_uint32(self.observations_timestamp)?);
buffer.extend_from_slice(&ReportBase::encode_uint192(&self.native_fee)?);
buffer.extend_from_slice(&ReportBase::encode_uint192(&self.link_fee)?);
buffer.extend_from_slice(&ReportBase::encode_uint32(self.expires_at)?);
buffer.extend_from_slice(&ReportBase::encode_uint64(self.last_update_timestamp)?);
buffer.extend_from_slice(&ReportBase::encode_int192(&self.mid_price)?);
buffer.extend_from_slice(&ReportBase::encode_uint32(self.market_status)?);
Ok(buffer)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::report::tests::{
generate_mock_report_data_v8, MOCK_FEE, MOCK_PRICE, MOCK_TIMESTAMP, MARKET_STATUS_OPEN
};
const V8_FEED_ID_STR: &str =
"0x00086b4aa7e57ca7b68ae1bf45653f56b656fd3aa335ef7fae696b663f1b8472";
#[test]
fn test_decode_report_data_v8() {
let report_data = generate_mock_report_data_v8();
let encoded = report_data.abi_encode().unwrap();
let decoded = ReportDataV8::decode(&encoded).unwrap();
let expected_feed_id = ID::from_hex_str(V8_FEED_ID_STR).unwrap();
let expected_timestamp: u32 = MOCK_TIMESTAMP;
let expected_fee = BigInt::from(MOCK_FEE);
let expected_price = BigInt::from(MOCK_PRICE);
let expected_market_status: u32 = MARKET_STATUS_OPEN;
assert_eq!(decoded.feed_id, expected_feed_id);
assert_eq!(decoded.valid_from_timestamp, expected_timestamp);
assert_eq!(decoded.observations_timestamp, expected_timestamp);
assert_eq!(decoded.native_fee, expected_fee);
assert_eq!(decoded.link_fee, expected_fee);
assert_eq!(decoded.expires_at, expected_timestamp + 100);
assert_eq!(decoded.last_update_timestamp, expected_timestamp as u64);
assert_eq!(decoded.mid_price, expected_price);
assert_eq!(decoded.market_status, expected_market_status);
}
}