use fastcrypto::encoding::Base58;
use fastcrypto::encoding::Base64;
use move_core_types::annotated_value::MoveDatatypeLayout;
use move_core_types::identifier::Identifier;
use move_core_types::language_storage::StructTag;
use mysten_metrics::monitored_scope;
use schemars::JsonSchema;
use serde::{Deserialize, Serialize};
use serde_json::{json, Value};
use serde_with::{serde_as, DisplayFromStr};
use std::fmt;
use std::fmt::Display;
use sui_types::base_types::{ObjectID, SuiAddress, TransactionDigest};
use sui_types::error::SuiResult;
use sui_types::event::{Event, EventEnvelope, EventID};
use sui_types::sui_serde::BigInt;
use json_to_table::json_to_table;
use tabled::settings::Style as TableStyle;
use crate::{type_and_fields_from_move_event_data, Page};
use sui_types::sui_serde::SuiStructTag;
use std::str::FromStr;
pub type EventPage = Page<SuiEvent, EventID>;
#[serde_as]
#[derive(Eq, PartialEq, Clone, Debug, Serialize, Deserialize, JsonSchema)]
#[serde(rename = "Event", rename_all = "camelCase")]
pub struct SuiEvent {
pub id: EventID,
pub package_id: ObjectID,
#[schemars(with = "String")]
#[serde_as(as = "DisplayFromStr")]
pub transaction_module: Identifier,
pub sender: SuiAddress,
#[schemars(with = "String")]
#[serde_as(as = "SuiStructTag")]
pub type_: StructTag,
pub parsed_json: Value,
#[serde(flatten)]
pub bcs: BcsEvent,
#[serde(skip_serializing_if = "Option::is_none")]
#[schemars(with = "Option<BigInt<u64>>")]
#[serde_as(as = "Option<BigInt<u64>>")]
pub timestamp_ms: Option<u64>,
}
#[serde_as]
#[derive(Eq, PartialEq, Clone, Debug, Serialize, Deserialize, JsonSchema)]
#[serde(rename_all = "camelCase", tag = "bcsEncoding")]
#[serde(from = "MaybeTaggedBcsEvent")]
pub enum BcsEvent {
Base64 {
#[serde_as(as = "Base64")]
#[schemars(with = "Base64")]
bcs: Vec<u8>,
},
Base58 {
#[serde_as(as = "Base58")]
#[schemars(with = "Base58")]
bcs: Vec<u8>,
},
}
impl BcsEvent {
pub fn new(bytes: Vec<u8>) -> Self {
Self::Base64 { bcs: bytes }
}
pub fn bytes(&self) -> &[u8] {
match self {
BcsEvent::Base64 { bcs } => bcs.as_ref(),
BcsEvent::Base58 { bcs } => bcs.as_ref(),
}
}
pub fn into_bytes(self) -> Vec<u8> {
match self {
BcsEvent::Base64 { bcs } => bcs,
BcsEvent::Base58 { bcs } => bcs,
}
}
}
#[allow(unused)]
#[serde_as]
#[derive(Serialize, Deserialize)]
#[serde(rename_all = "camelCase", untagged)]
enum MaybeTaggedBcsEvent {
Tagged(TaggedBcsEvent),
Base58 {
#[serde_as(as = "Base58")]
bcs: Vec<u8>,
},
}
#[serde_as]
#[derive(Serialize, Deserialize)]
#[serde(rename_all = "camelCase", tag = "bcsEncoding")]
enum TaggedBcsEvent {
Base64 {
#[serde_as(as = "Base64")]
bcs: Vec<u8>,
},
Base58 {
#[serde_as(as = "Base58")]
bcs: Vec<u8>,
},
}
impl From<MaybeTaggedBcsEvent> for BcsEvent {
fn from(event: MaybeTaggedBcsEvent) -> BcsEvent {
let bcs = match event {
MaybeTaggedBcsEvent::Tagged(TaggedBcsEvent::Base58 { bcs })
| MaybeTaggedBcsEvent::Base58 { bcs } => bcs,
MaybeTaggedBcsEvent::Tagged(TaggedBcsEvent::Base64 { bcs }) => bcs,
};
Self::Base64 { bcs }
}
}
impl From<EventEnvelope> for SuiEvent {
fn from(ev: EventEnvelope) -> Self {
Self {
id: EventID {
tx_digest: ev.tx_digest,
event_seq: ev.event_num,
},
package_id: ev.event.package_id,
transaction_module: ev.event.transaction_module,
sender: ev.event.sender,
type_: ev.event.type_,
parsed_json: ev.parsed_json,
bcs: BcsEvent::Base64 {
bcs: ev.event.contents,
},
timestamp_ms: Some(ev.timestamp),
}
}
}
impl From<SuiEvent> for Event {
fn from(val: SuiEvent) -> Self {
Event {
package_id: val.package_id,
transaction_module: val.transaction_module,
sender: val.sender,
type_: val.type_,
contents: val.bcs.into_bytes(),
}
}
}
impl SuiEvent {
pub fn try_from(
event: Event,
tx_digest: TransactionDigest,
event_seq: u64,
timestamp_ms: Option<u64>,
layout: MoveDatatypeLayout,
) -> SuiResult<Self> {
let Event {
package_id,
transaction_module,
sender,
type_: _,
contents,
} = event;
let bcs = BcsEvent::Base64 {
bcs: contents.to_vec(),
};
let move_value = Event::move_event_to_move_value(&contents, layout)?;
let (type_, fields) = type_and_fields_from_move_event_data(move_value)?;
Ok(SuiEvent {
id: EventID {
tx_digest,
event_seq,
},
package_id,
transaction_module,
sender,
type_,
parsed_json: fields,
bcs,
timestamp_ms,
})
}
}
impl Display for SuiEvent {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let parsed_json = &mut self.parsed_json.clone();
bytes_array_to_base64(parsed_json);
let mut table = json_to_table(parsed_json);
let style = TableStyle::modern();
table.collapse().with(style);
write!(f,
" ┌──\n │ EventID: {}:{}\n │ PackageID: {}\n │ Transaction Module: {}\n │ Sender: {}\n │ EventType: {}\n",
self.id.tx_digest, self.id.event_seq, self.package_id, self.transaction_module, self.sender, self.type_)?;
if let Some(ts) = self.timestamp_ms {
writeln!(f, " │ Timestamp: {}\n └──", ts)?;
}
writeln!(f, " │ ParsedJSON:")?;
let table_string = table.to_string();
let table_rows = table_string.split_inclusive('\n');
for r in table_rows {
write!(f, " │ {r}")?;
}
write!(f, "\n └──")
}
}
impl SuiEvent {
pub fn random_for_testing() -> Self {
Self {
id: EventID {
tx_digest: TransactionDigest::random(),
event_seq: 0,
},
package_id: ObjectID::random(),
transaction_module: Identifier::from_str("random_for_testing").unwrap(),
sender: SuiAddress::random_for_testing_only(),
type_: StructTag::from_str("0x6666::random_for_testing::RandomForTesting").unwrap(),
parsed_json: json!({}),
bcs: BcsEvent::new(vec![]),
timestamp_ms: None,
}
}
}
fn bytes_array_to_base64(v: &mut Value) {
match v {
Value::Null | Value::Bool(_) | Value::Number(_) | Value::String(_) => (),
Value::Array(vals) => {
if let Some(vals) = vals.iter().map(try_into_byte).collect::<Option<Vec<_>>>() {
*v = json!(Base64::from_bytes(&vals).encoded())
} else {
for val in vals {
bytes_array_to_base64(val)
}
}
}
Value::Object(map) => {
for val in map.values_mut() {
bytes_array_to_base64(val)
}
}
}
}
fn try_into_byte(v: &Value) -> Option<u8> {
let num = v.as_u64()?;
(num <= 255).then_some(num as u8)
}
#[serde_as]
#[derive(Clone, Debug, Serialize, Deserialize, JsonSchema)]
pub enum EventFilter {
All([Box<EventFilter>; 0]),
Any(Vec<EventFilter>),
Sender(SuiAddress),
Transaction(
TransactionDigest,
),
MoveModule {
package: ObjectID,
#[schemars(with = "String")]
#[serde_as(as = "DisplayFromStr")]
module: Identifier,
},
MoveEventType(
#[schemars(with = "String")]
#[serde_as(as = "SuiStructTag")]
StructTag,
),
MoveEventModule {
package: ObjectID,
#[schemars(with = "String")]
#[serde_as(as = "DisplayFromStr")]
module: Identifier,
},
#[serde(rename_all = "camelCase")]
TimeRange {
#[schemars(with = "BigInt<u64>")]
#[serde_as(as = "BigInt<u64>")]
start_time: u64,
#[schemars(with = "BigInt<u64>")]
#[serde_as(as = "BigInt<u64>")]
end_time: u64,
},
}
impl Filter<SuiEvent> for EventFilter {
fn matches(&self, item: &SuiEvent) -> bool {
let _scope = monitored_scope("EventFilter::matches");
match self {
EventFilter::All([]) => true,
EventFilter::Any(filters) => filters.iter().any(|f| f.matches(item)),
EventFilter::MoveEventType(event_type) => &item.type_ == event_type,
EventFilter::Sender(sender) => &item.sender == sender,
EventFilter::MoveModule { package, module } => {
&item.transaction_module == module && &item.package_id == package
}
EventFilter::Transaction(digest) => digest == &item.id.tx_digest,
EventFilter::TimeRange {
start_time,
end_time,
} => {
if let Some(timestamp) = &item.timestamp_ms {
start_time <= timestamp && end_time > timestamp
} else {
false
}
}
EventFilter::MoveEventModule { package, module } => {
&item.type_.module == module && &ObjectID::from(item.type_.address) == package
}
}
}
}
pub trait Filter<T> {
fn matches(&self, item: &T) -> bool;
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn bcs_event_test() {
let bytes = vec![0, 1, 2, 3, 4];
let untagged_base58 = r#"{"bcs":"12VfUX"}"#;
let tagged_base58 = r#"{"bcsEncoding":"base58","bcs":"12VfUX"}"#;
let tagged_base64 = r#"{"bcsEncoding":"base64","bcs":"AAECAwQ="}"#;
assert_eq!(
bytes,
serde_json::from_str::<BcsEvent>(untagged_base58)
.unwrap()
.into_bytes()
);
assert_eq!(
bytes,
serde_json::from_str::<BcsEvent>(tagged_base58)
.unwrap()
.into_bytes()
);
assert_eq!(
bytes,
serde_json::from_str::<BcsEvent>(tagged_base64)
.unwrap()
.into_bytes()
);
let event = serde_json::from_str::<BcsEvent>(tagged_base64).unwrap();
let json = serde_json::to_string(&event).unwrap();
let from_json = serde_json::from_str::<BcsEvent>(&json).unwrap();
assert_eq!(event, from_json);
let event = serde_json::from_str::<BcsEvent>(tagged_base58).unwrap();
let json = serde_json::to_string(&event).unwrap();
let from_json = serde_json::from_str::<BcsEvent>(&json).unwrap();
assert_eq!(event, from_json);
}
}