-
Notifications
You must be signed in to change notification settings - Fork 189
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat(sozo): support multicall for execute command #2897
Conversation
WalkthroughOhayo, sensei! The pull request introduces significant modifications to the Changes
Possibly related issues
Possibly related PRs
Suggested labels
🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments. CodeRabbit Commands (Invoked using PR comments)
Other keywords and placeholders
Documentation and Community
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 2
🧹 Nitpick comments (1)
bin/sozo/src/commands/execute.rs (1)
190-215
: Ohayo, sensei! Consider adding more test cases forCallArguments::from_str
The current tests cover several cases, but you might want to add tests for invalid calldata formats to ensure robust parsing and error handling.
Here's an example of an additional test case:
// invalid calldata format let res = CallArguments::from_str("0x1234,run,invalid_calldata"); assert!(res.is_err());
📜 Review details
Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (2)
bin/sozo/src/commands/execute.rs
(5 hunks)crates/sozo/ops/src/resource_descriptor.rs
(1 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (2)
- GitHub Check: build
- GitHub Check: clippy
🔇 Additional comments (1)
crates/sozo/ops/src/resource_descriptor.rs (1)
10-10
: Ohayo, sensei! Nice addition ofPartialEq
derive forResourceDescriptor
Deriving
PartialEq
allows for equality comparisons, which is useful in testing and other logical operations.
if self.diff { | ||
message.push_str( | ||
" Run the command again with `--diff` to force the fetch of data from \ | ||
the chain.", | ||
); | ||
} | ||
anyhow!(message) | ||
})?; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ohayo, sensei! Fix the error message condition to display the suggestion correctly
Currently, the error message suggests running the command again with --diff
when self.diff
is true
. However, if self.diff
is already true
, the suggestion is redundant. The condition should be inverted to check when self.diff
is false
.
Apply this diff to fix the condition:
- if self.diff {
+ if !self.diff {
📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.
if self.diff { | |
message.push_str( | |
" Run the command again with `--diff` to force the fetch of data from \ | |
the chain.", | |
); | |
} | |
anyhow!(message) | |
})?; | |
if !self.diff { | |
message.push_str( | |
" Run the command again with `--diff` to force the fetch of data from \ | |
the chain.", | |
); | |
} | |
anyhow!(message) | |
})?; |
let contract_address = match &descriptor { | ||
ResourceDescriptor::Address(address) => Some(*address), | ||
ResourceDescriptor::Tag(tag) => contracts.get(tag).map(|c| c.address), | ||
ResourceDescriptor::Name(_) => { | ||
unimplemented!("Expected to be a resolved tag with default namespace.") | ||
} | ||
}; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Ohayo, sensei! Handle ResourceDescriptor::Name
variant to prevent runtime panics
Using unimplemented!()
in the match arm for ResourceDescriptor::Name(_)
will cause a runtime panic if this case is encountered. Consider implementing proper error handling for this variant to prevent unexpected panics.
Apply this diff to handle the Name
variant gracefully:
ResourceDescriptor::Name(_) => {
- unimplemented!("Expected to be a resolved tag with default namespace.")
+ return Err(anyhow!("Contract address could not be resolved for descriptor: {descriptor}"));
}
Committable suggestion skipped: line range outside the PR's diff.
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## main #2897 +/- ##
==========================================
- Coverage 55.85% 55.20% -0.65%
==========================================
Files 449 449
Lines 57730 58350 +620
==========================================
- Hits 32245 32215 -30
- Misses 25485 26135 +650 ☔ View full report in Codecov by Sentry. |
bin/sozo/src/commands/execute.rs
Outdated
pub tag_or_address: ResourceDescriptor, | ||
#[arg(num_args = 1..)] | ||
#[arg(help = "A list of calls to execute.\n | ||
A call is made up of 3 values, separated by a comma (<TAG_OR_ADDRESS>,<ENTRYPOINT>[,<CALLDATA>]): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
May be more natural to have the calldata separated from the two first arguments.
Starkli is doing that: /~https://github.com/xJonathanLEI/starkli/blob/4c92ac0ee3c8a50289b09760ec535cf27dd64b2c/src/subcommands/invoke.rs#L62
Wondering what could be best to have something easy to use, wdyt on that?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 0
♻️ Duplicate comments (2)
bin/sozo/src/commands/execute.rs (2)
115-117
:⚠️ Potential issueHandle ResourceDescriptor::Name variant to prevent runtime panics.
Ohayo, sensei! Using
unimplemented!()
could cause runtime panics. We should handle this case gracefully.ResourceDescriptor::Name(_) => { - unimplemented!("Expected to be a resolved tag with default namespace.") + return Err(anyhow!("Contract address could not be resolved for descriptor: {descriptor}")); }
122-129
:⚠️ Potential issueFix the error message condition to display the suggestion correctly.
Ohayo, sensei! The suggestion to use
--diff
should only be shown when it's not already being used.- if self.diff { + if !self.diff {
🧹 Nitpick comments (2)
bin/sozo/src/commands/execute.rs (2)
137-137
: Define call separator characters as constants.Ohayo, sensei! The separator characters should be defined as constants at the module level for better maintainability.
+const CALL_SEPARATORS: [&str; 3] = ["/", "-", "\\"]; // Then in the code: - "/" | "-" | "\\" => break, + s if CALL_SEPARATORS.contains(&s) => break,
131-132
: Improve error message for missing entrypoint.Ohayo, sensei! The error message could be more descriptive about what argument is missing.
- arg_iter.next().ok_or_else(|| anyhow!("Unexpected number of arguments"))?; + arg_iter.next().ok_or_else(|| anyhow!("Missing entrypoint for contract {descriptor}"))?;
📜 Review details
Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (3)
bin/sozo/src/commands/execute.rs
(2 hunks)bin/sozo/src/commands/mod.rs
(1 hunks)crates/dojo/world/src/config/calldata_decoder.rs
(2 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (1)
- GitHub Check: fmt
🔇 Additional comments (5)
crates/dojo/world/src/config/calldata_decoder.rs (1)
143-143
: LGTM! Clean refactoring of the decoder function.The renaming from
decode_inner
todecode_single_calldata
and making it public improves code clarity and reusability. The function's implementation remains solid with comprehensive test coverage.Also applies to: 157-157
bin/sozo/src/commands/mod.rs (1)
49-52
: LGTM! Documentation updates accurately reflect the new multicall capability.The command descriptions have been updated to clearly indicate that the Execute command now supports multiple system calls.
Also applies to: 54-54
bin/sozo/src/commands/execute.rs (3)
23-24
: Add the required constraint before num_args for proper validation.Ohayo, sensei! Based on the past review comments, the argument constraints need to be reordered for proper application.
- #[arg(num_args = 1..)] - #[arg(required = true)] + #[arg(required = true)] + #[arg(num_args = 1..)]
90-97
: LGTM! Clean implementation of contract resolution.The contract resolution logic properly handles both manifest-based and diff-based approaches.
150-154
: LGTM! Proper multicall implementation.The implementation correctly builds and executes multiple calls in a single transaction.
5ee7a30
to
09b00d1
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 1
♻️ Duplicate comments (2)
bin/sozo/src/commands/execute.rs (2)
115-118
:⚠️ Potential issueOhayo, sensei! Replace unimplemented!() with proper error handling
Using
unimplemented!()
for theResourceDescriptor::Name
variant could cause runtime panics.Apply this diff to handle the Name variant gracefully:
ResourceDescriptor::Name(_) => { - unimplemented!("Expected to be a resolved tag with default namespace.") + return Err(anyhow!("Contract address could not be resolved for descriptor: {descriptor}")); }
122-129
:⚠️ Potential issueOhayo, sensei! Fix the error message condition
The error message suggests running with
--diff
when it's already enabled.Apply this diff to fix the condition:
- if self.diff { + if !self.diff {
🧹 Nitpick comments (2)
bin/sozo/src/commands/execute.rs (2)
157-157
: Ohayo, sensei! Consider adding transaction simulationThe multicall transaction could benefit from simulation before execution to catch potential issues early.
Consider adding a simulation step before the actual multicall:
+ // Simulate the transaction first + let simulation = invoker.simulate_multicall().await?; + trace!(simulation=?simulation, "Multicall simulation successful"); + let tx_result = invoker.multicall().await?;
104-155
: Ohayo, sensei! The multicall implementation is well-structuredThe implementation efficiently batches multiple calls into a single transaction, which is a significant improvement for gas efficiency and atomic execution. The parsing logic for calls and calldata is robust and handles edge cases well.
Consider documenting the following characteristics in the codebase:
- Gas savings from batching multiple calls
- Atomicity guarantees (all calls succeed or all fail)
- Maximum recommended number of calls per multicall
📜 Review details
Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (3)
bin/sozo/src/commands/execute.rs
(2 hunks)bin/sozo/src/commands/mod.rs
(1 hunks)crates/dojo/world/src/config/calldata_decoder.rs
(2 hunks)
🚧 Files skipped from review as they are similar to previous changes (2)
- bin/sozo/src/commands/mod.rs
- crates/dojo/world/src/config/calldata_decoder.rs
⏰ Context from checks skipped due to timeout of 90000ms (1)
- GitHub Check: clippy
🔇 Additional comments (1)
bin/sozo/src/commands/execute.rs (1)
21-51
: Ohayo, sensei! The command structure is well-documented and properly implemented.The help text is comprehensive, providing clear examples and detailed explanations of the call format and supported type prefixes. The implementation correctly follows the previous feedback about argument constraints.
let mut arg_iter = self.calls.into_iter(); | ||
|
||
while let Some(arg) = arg_iter.next() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Ohayo, sensei! Add validation for minimum number of calls
The iterator implementation should validate that at least one complete call (address + entrypoint) is provided.
Add this validation before the while loop:
let mut arg_iter = self.calls.into_iter();
+ if self.calls.len() < 2 {
+ return Err(anyhow!("At least one complete call (address and entrypoint) is required"));
+ }
while let Some(arg) = arg_iter.next() {
📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.
let mut arg_iter = self.calls.into_iter(); | |
while let Some(arg) = arg_iter.next() { | |
let mut arg_iter = self.calls.into_iter(); | |
if self.calls.len() < 2 { | |
return Err(anyhow!("At least one complete call (address and entrypoint) is required")); | |
} | |
while let Some(arg) = arg_iter.next() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you @remybar! One small change and let's roll.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 1
🧹 Nitpick comments (2)
bin/sozo/src/commands/execute.rs (2)
21-51
: Ohayo, sensei! Consider enhancing the documentation with more comprehensive examplesWhile the documentation is thorough, consider adding examples that demonstrate:
- All supported prefixes (u256, sstr, str, int)
- Multiple calls with different calldata types
- The maximum number of calls supported in a single transaction
Add an expanded example section:
EXAMPLE sozo execute 0x1234 run / ns-Actions move 1 2 + + # Multiple calls with different data types + sozo execute \ + 0x1234 set_value u256:1000 / \ + ns-Actions move int:42 sstr:player / \ + ns-Config update_name str:ByteArrayValue
155-160
: Ohayo, sensei! Consider adding debug assertions for call validationAdd debug assertions to validate each call's parameters before adding to the multicall.
+ debug_assert!(!calldata.is_empty(), "Calldata should not be empty"); + debug_assert!( + contract_address != starknet::core::types::FieldElement::ZERO, + "Contract address should not be zero" + ); invoker.add_call(Call { to: contract_address, selector: snutils::get_selector_from_name(&entrypoint)?, calldata, });
📜 Review details
Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (1)
bin/sozo/src/commands/execute.rs
(2 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (1)
- GitHub Check: fmt
🔇 Additional comments (3)
bin/sozo/src/commands/execute.rs (3)
123-130
: Ohayo, sensei! Fix the error message conditionThe error message suggests running with
--diff
when it's already true.Apply this diff to fix the condition:
- if self.diff { + if !self.diff { message.push_str( " Run the command again with `--diff` to force the fetch of data from \ the chain.", ); }
116-119
: Ohayo, sensei! Handle ResourceDescriptor::Name variant properlyUsing
unimplemented!()
will cause runtime panics. Handle the Name variant gracefully:ResourceDescriptor::Name(_) => { - unimplemented!("Expected to be a resolved tag with default namespace.") + return Err(anyhow!("Contract address could not be resolved for descriptor: {descriptor}")); }
106-107
:⚠️ Potential issueOhayo, sensei! Add validation for the number of calls
The iterator setup should validate both minimum and maximum number of calls to prevent:
- Empty or incomplete calls
- Exceeding Starknet's transaction size limits
Add validation before the iterator setup:
+ const MAX_CALLS: usize = 100; // Adjust based on Starknet's limits + if self.calls.len() < 2 { + return Err(anyhow!("At least one complete call (address and entrypoint) is required")); + } + if self.calls.len() > MAX_CALLS * 3 { + return Err(anyhow!("Number of calls exceeds maximum limit")); + } + let mut arg_iter = self.calls.into_iter();Likely invalid or redundant comment.
let mut calldata = vec![]; | ||
for arg in &mut arg_iter { | ||
let arg = match arg.as_str() { | ||
"/" | "-" | "\\" => break, | ||
_ => calldata_decoder::decode_single_calldata(&arg)?, | ||
}; | ||
calldata.extend(arg); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ohayo, sensei! Add safety check for calldata parsing
The calldata parsing loop could potentially continue indefinitely if a separator is missing. Additionally, there's no validation of calldata size.
Add safety checks:
let mut calldata = vec![];
+ let mut calldata_count = 0;
+ const MAX_CALLDATA_SIZE: usize = 1000; // Adjust based on requirements
for arg in &mut arg_iter {
+ calldata_count += 1;
+ if calldata_count > MAX_CALLDATA_SIZE {
+ return Err(anyhow!("Calldata size exceeds maximum limit"));
+ }
let arg = match arg.as_str() {
"/" | "-" | "\\" => break,
_ => calldata_decoder::decode_single_calldata(&arg)?,
};
calldata.extend(arg);
}
📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.
let mut calldata = vec![]; | |
for arg in &mut arg_iter { | |
let arg = match arg.as_str() { | |
"/" | "-" | "\\" => break, | |
_ => calldata_decoder::decode_single_calldata(&arg)?, | |
}; | |
calldata.extend(arg); | |
} | |
let mut calldata = vec![]; | |
let mut calldata_count = 0; | |
const MAX_CALLDATA_SIZE: usize = 1000; // Adjust based on requirements | |
for arg in &mut arg_iter { | |
calldata_count += 1; | |
if calldata_count > MAX_CALLDATA_SIZE { | |
return Err(anyhow!("Calldata size exceeds maximum limit")); | |
} | |
let arg = match arg.as_str() { | |
"/" | "-" | "\\" => break, | |
_ => calldata_decoder::decode_single_calldata(&arg)?, | |
}; | |
calldata.extend(arg); | |
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Awesome, thank you @remybar!
As we discussed, if you want to add the support for array of single felt values, could be a great addition too. 👍
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 1
♻️ Duplicate comments (1)
bin/sozo/src/commands/execute.rs (1)
139-146
:⚠️ Potential issueOhayo, sensei! Add safety checks for calldata parsing
The calldata parsing loop needs safety checks for size limits and proper termination.
Applying the previously suggested fix with a minor improvement:
let mut calldata = vec![]; + let mut calldata_count = 0; + const MAX_CALLDATA_SIZE: usize = 1000; // Adjust based on requirements for arg in &mut arg_iter { + if !matches!(arg.as_str(), "/" | "-" | "\\") { + calldata_count += 1; + if calldata_count > MAX_CALLDATA_SIZE { + return Err(anyhow!("Calldata size exceeds maximum limit of {MAX_CALLDATA_SIZE} for contract {tag_or_address}")); + } + } let arg = match arg.as_str() { "/" | "-" | "\\" => break, _ => calldata_decoder::decode_single_calldata(&arg)?, }; calldata.extend(arg); }
🧹 Nitpick comments (2)
bin/sozo/src/commands/execute.rs (2)
21-51
: Ohayo, sensei! The command structure looks good with clear documentationThe help text is comprehensive and includes good examples. The argument constraints are properly set following previous feedback.
Consider adding these clarifications to the help text:
- Mention that multiple calls will be executed in a single transaction
- Add a note about the order of execution for multiple calls
#[arg(help = "A list of calls to execute, separated by a /. + All calls will be executed in a single transaction in the order they are specified. A call is made up of a <TAG_OR_ADDRESS>, an <ENTRYPOINT> and an optional <CALLDATA>:
132-137
: Ohayo, sensei! Improve error message for missing entrypointThe error message could be more informative by including the expected format.
- "You must specify the entry point of the contract `{tag_or_address}` to \ - invoke, and optionally the calldata." + "Missing entrypoint for contract `{tag_or_address}`. Expected format: \ + <TAG_OR_ADDRESS> <ENTRYPOINT> [CALLDATA] [/ <NEXT_CALL>...]"
📜 Review details
Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (1)
bin/sozo/src/commands/execute.rs
(2 hunks)
🔇 Additional comments (2)
bin/sozo/src/commands/execute.rs (2)
155-160
: 🛠️ Refactor suggestionOhayo, sensei! Consider adding call validation before execution
Before adding each call to the multicall, we should validate the calldata size and format.
+ // Validate calldata size + if calldata.len() > MAX_CALLDATA_SIZE { + return Err(anyhow!( + "Total calldata size {} exceeds maximum limit {} for contract {}", + calldata.len(), + MAX_CALLDATA_SIZE, + tag_or_address + )); + } + invoker.add_call(Call { to: contract_address, selector: snutils::get_selector_from_name(&entrypoint)?, calldata, });Likely invalid or redundant comment.
106-107
: 🛠️ Refactor suggestionOhayo, sensei! Add validation for call format
The iterator implementation should validate the call format before processing.
Add this validation before the while loop:
let mut arg_iter = self.calls.into_iter(); + if self.calls.is_empty() { + return Err(anyhow!("At least one call (address and entrypoint) is required")); + } + if self.calls.len() == 1 { + return Err(anyhow!("Incomplete call: entrypoint is missing for the specified address")); + }Likely invalid or redundant comment.
}); | ||
} | ||
|
||
let tx_result = invoker.multicall().await?; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Ohayo, sensei! Add transaction size validation
Before executing the multicall, validate the total transaction size.
+ // Validate total transaction size
+ const MAX_TRANSACTION_SIZE: usize = 5000; // Adjust based on chain limits
+ let total_size = invoker.get_calls().iter().map(|call| call.calldata.len()).sum::<usize>();
+ if total_size > MAX_TRANSACTION_SIZE {
+ return Err(anyhow!(
+ "Total transaction size {} exceeds maximum limit {}",
+ total_size,
+ MAX_TRANSACTION_SIZE
+ ));
+ }
+
let tx_result = invoker.multicall().await?;
Committable suggestion skipped: line range outside the PR's diff.
Related issue
#2388
Tests
Added to documentation?
Checklist
scripts/prettier.sh
,scripts/rust_fmt.sh
,scripts/cairo_fmt.sh
)scripts/clippy.sh
,scripts/docs.sh
)Summary by CodeRabbit
Summary by CodeRabbit
New Features
Improvements
Technical Enhancements
Execute
command to clarify its functionality.