Skip to content

Commit

Permalink
feat(katana): rpc modules selection (#2848)
Browse files Browse the repository at this point in the history
rpc modules selection
  • Loading branch information
kariy committed Dec 31, 2024
1 parent 19b0486 commit 38b3c2a
Show file tree
Hide file tree
Showing 7 changed files with 199 additions and 30 deletions.
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

6 changes: 3 additions & 3 deletions crates/dojo/test-utils/src/sequencer.rs
Original file line number Diff line number Diff line change
@@ -1,16 +1,16 @@
use std::collections::HashSet;
use std::sync::Arc;

use katana_core::backend::Backend;
use katana_core::constants::DEFAULT_SEQUENCER_ADDRESS;
use katana_executor::implementation::blockifier::BlockifierFactory;
use katana_node::config::dev::DevConfig;
use katana_node::config::rpc::{ApiKind, RpcConfig, DEFAULT_RPC_ADDR, DEFAULT_RPC_MAX_CONNECTIONS};
use katana_node::config::rpc::{RpcConfig, DEFAULT_RPC_ADDR, DEFAULT_RPC_MAX_CONNECTIONS};
pub use katana_node::config::*;
use katana_node::LaunchedNode;
use katana_primitives::chain::ChainId;
use katana_primitives::chain_spec::ChainSpec;
use katana_rpc::Error;
use rpc::RpcModulesList;
use starknet::accounts::{ExecutionEncoding, SingleOwnerAccount};
use starknet::core::chain_id;
use starknet::core::types::{BlockId, BlockTag, Felt};
Expand Down Expand Up @@ -122,8 +122,8 @@ pub fn get_default_test_config(sequencing: SequencingConfig) -> Config {
cors_origins: Vec::new(),
port: 0,
addr: DEFAULT_RPC_ADDR,
apis: RpcModulesList::all(),
max_connections: DEFAULT_RPC_MAX_CONNECTIONS,
apis: HashSet::from([ApiKind::Starknet, ApiKind::Dev, ApiKind::Saya, ApiKind::Torii]),
max_event_page_size: Some(100),
max_proof_keys: Some(100),
};
Expand Down
61 changes: 47 additions & 14 deletions crates/katana/cli/src/args.rs
Original file line number Diff line number Diff line change
@@ -1,11 +1,10 @@
//! Katana node CLI options and configuration.
use std::collections::HashSet;
use std::path::PathBuf;
use std::sync::Arc;

use alloy_primitives::U256;
use anyhow::{Context, Result};
use anyhow::{bail, Context, Result};
use clap::Parser;
use katana_core::constants::DEFAULT_SEQUENCER_ADDRESS;
use katana_core::service::messaging::MessagingConfig;
Expand All @@ -14,7 +13,7 @@ use katana_node::config::dev::{DevConfig, FixedL1GasPriceConfig};
use katana_node::config::execution::ExecutionConfig;
use katana_node::config::fork::ForkingConfig;
use katana_node::config::metrics::MetricsConfig;
use katana_node::config::rpc::{ApiKind, RpcConfig};
use katana_node::config::rpc::{RpcConfig, RpcModuleKind, RpcModulesList};
use katana_node::config::{Config, SequencingConfig};
use katana_primitives::chain_spec::{self, ChainSpec};
use katana_primitives::genesis::allocation::DevAllocationsGenerator;
Expand Down Expand Up @@ -169,7 +168,7 @@ impl NodeArgs {

pub fn config(&self) -> Result<katana_node::config::Config> {
let db = self.db_config();
let rpc = self.rpc_config();
let rpc = self.rpc_config()?;
let dev = self.dev_config();
let chain = self.chain_spec()?;
let metrics = self.metrics_config();
Expand Down Expand Up @@ -197,29 +196,39 @@ impl NodeArgs {
SequencingConfig { block_time: self.block_time, no_mining: self.no_mining }
}

fn rpc_config(&self) -> RpcConfig {
let mut apis = HashSet::from([ApiKind::Starknet, ApiKind::Torii, ApiKind::Saya]);
// only enable `katana` API in dev mode
if self.development.dev {
apis.insert(ApiKind::Dev);
}
fn rpc_config(&self) -> Result<RpcConfig> {
let modules = if let Some(modules) = &self.server.http_modules {
// TODO: This check should be handled in the `katana-node` level. Right now if you
// instantiate katana programmatically, you can still add the dev module without
// enabling dev mode.
//
// We only allow the `dev` module in dev mode (ie `--dev` flag)
if !self.development.dev && modules.contains(&RpcModuleKind::Dev) {
bail!("The `dev` module can only be enabled in dev mode (ie `--dev` flag)")
}

modules.clone()
} else {
// Expose the default modules if none is specified.
RpcModulesList::default()
};

#[cfg(feature = "server")]
{
RpcConfig {
apis,
Ok(RpcConfig {
apis: modules,
port: self.server.http_port,
addr: self.server.http_addr,
max_connections: self.server.max_connections,
cors_origins: self.server.http_cors_origins.clone(),
max_event_page_size: Some(self.server.max_event_page_size),
max_proof_keys: Some(self.server.max_proof_keys),
}
})
}

#[cfg(not(feature = "server"))]
{
RpcConfig { apis, ..Default::default() }
Ok(RpcConfig { apis, ..Default::default() })
}
}

Expand Down Expand Up @@ -636,4 +645,28 @@ chain_id.Named = "Mainnet"
assert!(cors_origins.contains(&HeaderValue::from_static("http://localhost:3000")));
assert!(cors_origins.contains(&HeaderValue::from_static("https://example.com")));
}

#[test]
fn http_modules() {
// If the `--http.api` isn't specified, only starknet module will be exposed.
let config = NodeArgs::parse_from(["katana"]).config().unwrap();
let modules = config.rpc.apis;
assert_eq!(modules.len(), 1);
assert!(modules.contains(&RpcModuleKind::Starknet));

// If the `--http.api` is specified, only the ones in the list will be exposed.
let config = NodeArgs::parse_from(["katana", "--http.api", "saya,torii"]).config().unwrap();
let modules = config.rpc.apis;
assert_eq!(modules.len(), 2);
assert!(modules.contains(&RpcModuleKind::Saya));
assert!(modules.contains(&RpcModuleKind::Torii));

// Specifiying the dev module without enabling dev mode is forbidden.
let err =
NodeArgs::parse_from(["katana", "--http.api", "starknet,dev"]).config().unwrap_err();
assert!(
err.to_string()
.contains("The `dev` module can only be enabled in dev mode (ie `--dev` flag)")
);
}
}
11 changes: 9 additions & 2 deletions crates/katana/cli/src/options.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ use std::net::IpAddr;
use clap::Args;
use katana_node::config::execution::{DEFAULT_INVOCATION_MAX_STEPS, DEFAULT_VALIDATION_MAX_STEPS};
use katana_node::config::metrics::{DEFAULT_METRICS_ADDR, DEFAULT_METRICS_PORT};
use katana_node::config::rpc::DEFAULT_RPC_MAX_PROOF_KEYS;
use katana_node::config::rpc::{RpcModulesList, DEFAULT_RPC_MAX_PROOF_KEYS};
#[cfg(feature = "server")]
use katana_node::config::rpc::{
DEFAULT_RPC_ADDR, DEFAULT_RPC_MAX_CONNECTIONS, DEFAULT_RPC_MAX_EVENT_PAGE_SIZE,
Expand Down Expand Up @@ -97,6 +97,12 @@ pub struct ServerOptions {
)]
pub http_cors_origins: Vec<HeaderValue>,

/// API's offered over the HTTP-RPC interface.
#[arg(long = "http.api", value_name = "MODULES")]
#[arg(value_parser = RpcModulesList::parse)]
#[serde(default)]
pub http_modules: Option<RpcModulesList>,

/// Maximum number of concurrent connections allowed.
#[arg(long = "rpc.max-connections", value_name = "COUNT")]
#[arg(default_value_t = DEFAULT_RPC_MAX_CONNECTIONS)]
Expand All @@ -122,8 +128,9 @@ impl Default for ServerOptions {
ServerOptions {
http_addr: DEFAULT_RPC_ADDR,
http_port: DEFAULT_RPC_PORT,
max_connections: DEFAULT_RPC_MAX_CONNECTIONS,
http_cors_origins: Vec::new(),
http_modules: Some(RpcModulesList::default()),
max_connections: DEFAULT_RPC_MAX_CONNECTIONS,
max_event_page_size: DEFAULT_RPC_MAX_EVENT_PAGE_SIZE,
max_proof_keys: DEFAULT_RPC_MAX_PROOF_KEYS,
}
Expand Down
1 change: 1 addition & 0 deletions crates/katana/node/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ jsonrpsee.workspace = true
serde.workspace = true
serde_json.workspace = true
starknet.workspace = true
thiserror.workspace = true
tower = { workspace = true, features = [ "full" ] }
tower-http = { workspace = true, features = [ "full" ] }
tracing.workspace = true
Expand Down
139 changes: 133 additions & 6 deletions crates/katana/node/src/config/rpc.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ use std::collections::HashSet;
use std::net::{IpAddr, Ipv4Addr, SocketAddr};

use katana_rpc::cors::HeaderValue;
use serde::{Deserialize, Serialize};

/// The default maximum number of concurrent RPC connections.
pub const DEFAULT_RPC_MAX_CONNECTIONS: u32 = 100;
Expand All @@ -13,15 +14,25 @@ pub const DEFAULT_RPC_MAX_EVENT_PAGE_SIZE: u64 = 1024;
/// Default maximmum number of keys for the `starknet_getStorageProof` RPC method.
pub const DEFAULT_RPC_MAX_PROOF_KEYS: u64 = 100;

/// List of APIs supported by Katana.
/// List of RPC modules supported by Katana.
#[derive(
Debug, Copy, Clone, PartialEq, Eq, Hash, strum_macros::EnumString, strum_macros::Display,
Debug,
Copy,
Clone,
PartialEq,
Eq,
Hash,
strum_macros::EnumString,
strum_macros::Display,
Serialize,
Deserialize,
)]
pub enum ApiKind {
#[strum(ascii_case_insensitive)]
pub enum RpcModuleKind {
Starknet,
Torii,
Dev,
Saya,
Dev,
}

/// Configuration for the RPC server.
Expand All @@ -30,7 +41,7 @@ pub struct RpcConfig {
pub addr: IpAddr,
pub port: u16,
pub max_connections: u32,
pub apis: HashSet<ApiKind>,
pub apis: RpcModulesList,
pub cors_origins: Vec<HeaderValue>,
pub max_event_page_size: Option<u64>,
pub max_proof_keys: Option<u64>,
Expand All @@ -49,10 +60,126 @@ impl Default for RpcConfig {
cors_origins: Vec::new(),
addr: DEFAULT_RPC_ADDR,
port: DEFAULT_RPC_PORT,
apis: RpcModulesList::default(),
max_connections: DEFAULT_RPC_MAX_CONNECTIONS,
apis: HashSet::from([ApiKind::Starknet]),
max_event_page_size: Some(DEFAULT_RPC_MAX_EVENT_PAGE_SIZE),
max_proof_keys: Some(DEFAULT_RPC_MAX_PROOF_KEYS),
}
}
}

#[derive(Debug, thiserror::Error)]
#[error("invalid module: {0}")]
pub struct InvalidRpcModuleError(String);

#[derive(Debug, Clone, PartialEq, Eq, Deserialize, Serialize)]
#[serde(transparent)]
pub struct RpcModulesList(HashSet<RpcModuleKind>);

impl RpcModulesList {
/// Creates an empty modules list.
pub fn new() -> Self {
Self(HashSet::new())
}

/// Creates a list with all the possible modules.
pub fn all() -> Self {
Self(HashSet::from([
RpcModuleKind::Starknet,
RpcModuleKind::Torii,
RpcModuleKind::Saya,
RpcModuleKind::Dev,
]))
}

/// Adds a `module` to the list.
pub fn add(&mut self, module: RpcModuleKind) {
self.0.insert(module);
}

/// Returns `true` if the list contains the specified `module`.
pub fn contains(&self, module: &RpcModuleKind) -> bool {
self.0.contains(module)
}

/// Returns the number of modules in the list.
pub fn len(&self) -> usize {
self.0.len()
}

/// Returns `true` if the list contains no modules.
pub fn is_empty(&self) -> bool {
self.0.is_empty()
}

/// Used as the value parser for `clap`.
pub fn parse(value: &str) -> Result<Self, InvalidRpcModuleError> {
if value.is_empty() {
return Ok(Self::new());
}

let mut modules = HashSet::new();
for module_str in value.split(',') {
let module: RpcModuleKind = module_str
.trim()
.parse()
.map_err(|_| InvalidRpcModuleError(module_str.to_string()))?;

modules.insert(module);
}

Ok(Self(modules))
}
}

impl Default for RpcModulesList {
fn default() -> Self {
Self(HashSet::from([RpcModuleKind::Starknet]))
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn test_parse_empty() {
let list = RpcModulesList::parse("").unwrap();
assert_eq!(list, RpcModulesList::new());
}

#[test]
fn test_parse_single() {
let list = RpcModulesList::parse("dev").unwrap();
assert!(list.contains(&RpcModuleKind::Dev));
}

#[test]
fn test_parse_multiple() {
let list = RpcModulesList::parse("dev,torii,saya").unwrap();
assert!(list.contains(&RpcModuleKind::Dev));
assert!(list.contains(&RpcModuleKind::Torii));
assert!(list.contains(&RpcModuleKind::Saya));
}

#[test]
fn test_parse_with_spaces() {
let list = RpcModulesList::parse(" dev , torii ").unwrap();
assert!(list.contains(&RpcModuleKind::Dev));
assert!(list.contains(&RpcModuleKind::Torii));
}

#[test]
fn test_parse_duplicates() {
let list = RpcModulesList::parse("dev,dev,torii").unwrap();
let mut expected = RpcModulesList::new();
expected.add(RpcModuleKind::Dev);
expected.add(RpcModuleKind::Torii);
assert_eq!(list, expected);
}

#[test]
fn test_parse_invalid() {
assert!(RpcModulesList::parse("invalid").is_err());
}
}
Loading

0 comments on commit 38b3c2a

Please sign in to comment.