diff --git a/crates/dojo-world/src/contracts/model.rs b/crates/dojo-world/src/contracts/model.rs index f6c1ec6c84..b2fee105da 100644 --- a/crates/dojo-world/src/contracts/model.rs +++ b/crates/dojo-world/src/contracts/model.rs @@ -1,5 +1,6 @@ use std::vec; +use async_trait::async_trait; use dojo_types::packing::{parse_ty, unpack, PackingError, ParseError}; use dojo_types::primitive::PrimitiveError; use dojo_types::schema::Ty; @@ -46,7 +47,17 @@ pub enum ModelError

{ Packing(#[from] PackingError), } -pub struct ModelReader<'a, P> { +#[cfg_attr(not(target_arch = "wasm32"), async_trait)] +#[cfg_attr(target_arch = "wasm32", async_trait(?Send))] +pub trait ModelReader { + fn class_hash(&self) -> FieldElement; + async fn schema(&self) -> Result; + async fn packed_size(&self) -> Result; + async fn unpacked_size(&self) -> Result; + async fn layout(&self) -> Result, E>; +} + +pub struct ModelRPCReader<'a, P: Sync + Send> { /// The name of the model name: FieldElement, /// The class hash of the model @@ -55,14 +66,14 @@ pub struct ModelReader<'a, P> { world_reader: &'a WorldContractReader

, } -impl<'a, P> ModelReader<'a, P> +impl<'a, P> ModelRPCReader<'a, P> where - P: Provider, + P: Provider + Sync + Send, { pub async fn new( name: &str, world: &'a WorldContractReader

, - ) -> Result, ModelError> { + ) -> Result, ModelError> { let name = cairo_short_string_to_felt(name)?; let class_hash = world @@ -88,11 +99,60 @@ where Ok(Self { world_reader: world, class_hash, name }) } - pub fn class_hash(&self) -> FieldElement { + pub async fn entity_storage( + &self, + keys: &[FieldElement], + ) -> Result, ModelError> { + let packed_size: u8 = + self.packed_size().await?.try_into().map_err(ParseError::ValueOutOfRange)?; + + let key = poseidon_hash_many(keys); + let key = poseidon_hash_many(&[short_string!("dojo_storage"), self.name, key]); + + let mut packed = Vec::with_capacity(packed_size as usize); + for slot in 0..packed_size { + let value = self + .world_reader + .provider() + .get_storage_at( + self.world_reader.address(), + key + slot.into(), + self.world_reader.block_id(), + ) + .await?; + + packed.push(value); + } + + Ok(packed) + } + + pub async fn entity(&self, keys: &[FieldElement]) -> Result> { + let mut schema = self.schema().await?; + + let layout = self.layout().await?; + let raw_values = self.entity_storage(keys).await?; + + let unpacked = unpack(raw_values, layout)?; + let mut keys_and_unpacked = [keys, &unpacked].concat(); + + schema.deserialize(&mut keys_and_unpacked)?; + + Ok(schema) + } +} + +#[cfg_attr(not(target_arch = "wasm32"), async_trait)] +#[cfg_attr(target_arch = "wasm32", async_trait(?Send))] +impl<'a, P> ModelReader> for ModelRPCReader<'a, P> +where + P: Provider + Sync + Send, +{ + fn class_hash(&self) -> FieldElement { self.class_hash } - pub async fn schema(&self) -> Result> { + async fn schema(&self) -> Result> { let entrypoint = get_selector_from_name(SCHEMA_SELECTOR_STR).unwrap(); let res = self @@ -103,7 +163,7 @@ where Ok(parse_ty(&res[1..])?) } - pub async fn packed_size(&self) -> Result> { + async fn packed_size(&self) -> Result> { let entrypoint = get_selector_from_name(PACKED_SIZE_SELECTOR_STR).unwrap(); let res = self @@ -114,7 +174,7 @@ where Ok(res[1]) } - pub async fn unpacked_size(&self) -> Result> { + async fn unpacked_size(&self) -> Result> { let entrypoint = get_selector_from_name(UNPACKED_SIZE_SELECTOR_STR).unwrap(); let res = self @@ -125,7 +185,7 @@ where Ok(res[1]) } - pub async fn layout(&self) -> Result, ModelError> { + async fn layout(&self) -> Result, ModelError> { let entrypoint = get_selector_from_name(LAYOUT_SELECTOR_STR).unwrap(); let res = self @@ -135,46 +195,4 @@ where Ok(res[2..].into()) } - - pub async fn entity_storage( - &self, - keys: &[FieldElement], - ) -> Result, ModelError> { - let packed_size: u8 = - self.packed_size().await?.try_into().map_err(ParseError::ValueOutOfRange)?; - - let key = poseidon_hash_many(keys); - let key = poseidon_hash_many(&[short_string!("dojo_storage"), self.name, key]); - - let mut packed = Vec::with_capacity(packed_size as usize); - for slot in 0..packed_size { - let value = self - .world_reader - .provider() - .get_storage_at( - self.world_reader.address(), - key + slot.into(), - self.world_reader.block_id(), - ) - .await?; - - packed.push(value); - } - - Ok(packed) - } - - pub async fn entity(&self, keys: &[FieldElement]) -> Result> { - let mut schema = self.schema().await?; - - let layout = self.layout().await?; - let raw_values = self.entity_storage(keys).await?; - - let unpacked = unpack(raw_values, layout)?; - let mut keys_and_unpacked = [keys, &unpacked].concat(); - - schema.deserialize(&mut keys_and_unpacked)?; - - Ok(schema) - } } diff --git a/crates/dojo-world/src/contracts/model_test.rs b/crates/dojo-world/src/contracts/model_test.rs index 300ac6ede2..6b4b5e1981 100644 --- a/crates/dojo-world/src/contracts/model_test.rs +++ b/crates/dojo-world/src/contracts/model_test.rs @@ -7,6 +7,7 @@ use dojo_types::schema::{Enum, EnumOption, Member, Struct, Ty}; use starknet::accounts::ConnectedAccount; use starknet::core::types::FieldElement; +use crate::contracts::model::ModelReader; use crate::contracts::world::test::deploy_world; use crate::contracts::world::WorldContractReader; diff --git a/crates/dojo-world/src/contracts/world.rs b/crates/dojo-world/src/contracts/world.rs index 1f9440ff7a..9421d97bb9 100644 --- a/crates/dojo-world/src/contracts/world.rs +++ b/crates/dojo-world/src/contracts/world.rs @@ -11,7 +11,7 @@ use starknet::core::utils::{ use starknet::macros::selector; use starknet::providers::{Provider, ProviderError}; -use super::model::{ModelError, ModelReader}; +use super::model::{ModelError, ModelRPCReader}; #[cfg(test)] #[path = "world_test.rs"] @@ -177,7 +177,7 @@ where pub async fn model( &'a self, name: &str, - ) -> Result, ModelError<::Error>> + ) -> Result, ModelError<::Error>> { self.reader.model(name).await } @@ -340,9 +340,12 @@ where impl<'a, P> WorldContractReader

where - P: Provider, + P: Provider + Sync + Send, { - pub async fn model(&'a self, name: &str) -> Result, ModelError> { - ModelReader::new(name, self).await + pub async fn model( + &'a self, + name: &str, + ) -> Result, ModelError> { + ModelRPCReader::new(name, self).await } } diff --git a/crates/sozo/src/ops/model.rs b/crates/sozo/src/ops/model.rs index aba6fa548d..c22404dcec 100644 --- a/crates/sozo/src/ops/model.rs +++ b/crates/sozo/src/ops/model.rs @@ -1,4 +1,5 @@ use anyhow::Result; +use dojo_world::contracts::model::ModelReader; use dojo_world::contracts::world::WorldContractReader; use dojo_world::metadata::Environment; use starknet::core::types::{BlockId, BlockTag}; diff --git a/crates/torii/core/src/processors/register_model.rs b/crates/torii/core/src/processors/register_model.rs index 2d63014c93..f632f6e437 100644 --- a/crates/torii/core/src/processors/register_model.rs +++ b/crates/torii/core/src/processors/register_model.rs @@ -1,5 +1,6 @@ use anyhow::{Error, Ok, Result}; use async_trait::async_trait; +use dojo_world::contracts::model::ModelReader; use dojo_world::contracts::world::WorldContractReader; use starknet::core::types::{BlockWithTxs, Event, InvokeTransactionReceipt}; use starknet::core::utils::parse_cairo_short_string;