Skip to content

Commit

Permalink
feat(katana): settlement layer initialization flow improvement (#2926)
Browse files Browse the repository at this point in the history
  • Loading branch information
kariy authored Jan 18, 2025
1 parent 74ccf13 commit 57cfe1e
Show file tree
Hide file tree
Showing 6 changed files with 400 additions and 121 deletions.
7 changes: 5 additions & 2 deletions Cargo.lock

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

7 changes: 5 additions & 2 deletions bin/katana/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,11 @@ repository.workspace = true
version.workspace = true

[dependencies]
katana-cairo.workspace = true
katana-cli.workspace = true
katana-db.workspace = true
katana-node.workspace = true
katana-primitives.workspace = true
katana-rpc-types.workspace = true

anyhow.workspace = true
byte-unit = "5.1.4"
Expand All @@ -23,16 +23,19 @@ comfy-table = "7.1.1"
dirs = "5.0.1"
dojo-utils.workspace = true
inquire = "0.7.5"
rand.workspace = true
serde.workspace = true
serde_json.workspace = true
shellexpand = "3.1.0"
spinoff.workspace = true
starknet.workspace = true
thiserror.workspace = true
tokio.workspace = true
toml.workspace = true
tracing.workspace = true

[dev-dependencies]
assert_matches.workspace = true
rstest.workspace = true
starknet.workspace = true

[features]
Expand Down
327 changes: 327 additions & 0 deletions bin/katana/src/cli/init/deployment.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,327 @@
use std::str::FromStr;
use std::sync::Arc;

use anyhow::{anyhow, Result};
use cainome::cairo_serde;
use cainome::rs::abigen;
use dojo_utils::{TransactionWaiter, TransactionWaitingError};
use katana_primitives::class::{
CompiledClassHash, ComputeClassHashError, ContractClass, ContractClassCompilationError,
ContractClassFromStrError,
};
use katana_primitives::{felt, ContractAddress, Felt};
use katana_rpc_types::class::RpcContractClass;
use spinoff::{spinners, Color, Spinner};
use starknet::accounts::{Account, AccountError, ConnectedAccount, SingleOwnerAccount};
use starknet::contract::ContractFactory;
use starknet::core::crypto::compute_hash_on_elements;
use starknet::core::types::{BlockId, BlockTag, FlattenedSierraClass, StarknetError};
use starknet::macros::short_string;
use starknet::providers::jsonrpc::HttpTransport;
use starknet::providers::{JsonRpcClient, Provider, ProviderError};
use starknet::signers::LocalWallet;
use thiserror::Error;
use tracing::trace;

type InitializerAccount = SingleOwnerAccount<Arc<JsonRpcClient<HttpTransport>>, LocalWallet>;

const PROGRAM_HASH: Felt =
felt!("0x5ab580b04e3532b6b18f81cfa654a05e29dd8e2352d88df1e765a84072db07");

/// The contract address that handles fact verification.
///
/// This address points to Herodotus' Atlantic Fact Registry contract on Starknet Sepolia as we rely
/// on their services to generates and verifies proofs.
const ATLANTIC_FACT_REGISTRY_SEPOLIA: Felt =
felt!("0x4ce7851f00b6c3289674841fd7a1b96b6fd41ed1edc248faccd672c26371b8c");

/// Deploys the settlement contract in the settlement layer and initializes it with the right
/// necessary states.
pub async fn deploy_settlement_contract(
mut account: InitializerAccount,
chain_id: Felt,
) -> Result<ContractAddress, ContractInitError> {
// This is important! Otherwise all the estimate fees after a transaction will be executed
// against invalid state.
account.set_block_id(BlockId::Tag(BlockTag::Pending));

let mut sp = Spinner::new(spinners::Dots, "", Color::Blue);

let result = async {
// -----------------------------------------------------------------------
// CONTRACT DEPLOYMENT
// -----------------------------------------------------------------------

let class = include_str!(
"../../../../../crates/katana/contracts/build/appchain_core_contract.json"
);

abigen!(
AppchainContract,
"[{\"type\":\"function\",\"name\":\"set_program_info\",\"inputs\":[{\"name\":\"\
program_hash\",\"type\":\"core::Felt\"},{\"name\":\"config_hash\",\"type\":\"\
core::Felt\"}],\"outputs\":[],\"state_mutability\":\"external\"},{\"type\":\"\
function\",\"name\":\"set_facts_registry\",\"inputs\":[{\"name\":\"address\",\"type\"\
:\"core::starknet::contract_address::ContractAddress\"}],\"outputs\":[],\"\
state_mutability\":\"external\"},{\"type\":\"function\",\"name\":\"\
get_facts_registry\",\"inputs\":[],\"outputs\":[{\"type\":\"\
core::starknet::contract_address::ContractAddress\"}],\"state_mutability\":\"view\"},\
{\"type\":\"function\",\"name\":\"get_program_info\",\"inputs\":[],\"outputs\":[{\"\
type\":\"(core::Felt, core::Felt)\"}],\"state_mutability\":\"view\"}]"
);

let class = ContractClass::from_str(class)?;
let class_hash = class.class_hash()?;

// Check if the class has already been declared,
match account.provider().get_class(BlockId::Tag(BlockTag::Pending), class_hash).await {
Ok(..) => {
// Class has already been declared, no need to do anything...
}

Err(ProviderError::StarknetError(StarknetError::ClassHashNotFound)) => {
sp.update_text("Declaring contract...");
let (rpc_class, casm_hash) = prepare_contract_declaration_params(class)?;

let res = account
.declare_v2(rpc_class.into(), casm_hash)
.send()
.await
.inspect(|res| {
let tx = format!("{:#x}", res.transaction_hash);
trace!(target: "init", %tx, "Transaction sent");
})
.map_err(ContractInitError::DeclarationError)?;

TransactionWaiter::new(res.transaction_hash, account.provider()).await?;
}

Err(err) => return Err(ContractInitError::Provider(err)),
}

sp.update_text("Deploying contract...");

let salt = Felt::from(rand::random::<u64>());
let factory = ContractFactory::new(class_hash, &account);

// appchain::constructor() /~https://github.com/cartridge-gg/piltover/blob/d373a844c3428383a48518adf468bf83249dec3a/src/appchain.cairo#L119-L125
let request = factory.deploy_v1(
vec![
account.address(), // owner
Felt::ZERO, // state_root
Felt::ZERO, // block_number
Felt::ZERO, // block_hash
],
salt,
false,
);

let res = request
.send()
.await
.inspect(|res| {
let tx = format!("{:#x}", res.transaction_hash);
trace!(target: "init", %tx, "Transaction sent");
})
.map_err(ContractInitError::DeploymentError)?;

TransactionWaiter::new(res.transaction_hash, account.provider()).await?;

// -----------------------------------------------------------------------
// CONTRACT INITIALIZATIONS
// -----------------------------------------------------------------------

let deployed_appchain_contract = request.deployed_address();
let appchain = AppchainContract::new(deployed_appchain_contract, &account);

// Compute the chain's config hash
let config_hash = compute_config_hash(
chain_id,
felt!("0x2e7442625bab778683501c0eadbc1ea17b3535da040a12ac7d281066e915eea"),
);

// 1. Program Info

sp.update_text("Setting program info...");

let res = appchain
.set_program_info(&PROGRAM_HASH, &config_hash)
.send()
.await
.inspect(|res| {
let tx = format!("{:#x}", res.transaction_hash);
trace!(target: "init", %tx, "Transaction sent");
})
.map_err(ContractInitError::Initialization)?;

TransactionWaiter::new(res.transaction_hash, account.provider()).await?;

// 2. Fact Registry

sp.update_text("Setting fact registry...");

let res = appchain
.set_facts_registry(&ATLANTIC_FACT_REGISTRY_SEPOLIA.into())
.send()
.await
.inspect(|res| {
let tx = format!("{:#x}", res.transaction_hash);
trace!(target: "init", %tx, "Transaction sent");
})
.map_err(ContractInitError::Initialization)?;

TransactionWaiter::new(res.transaction_hash, account.provider()).await?;

// -----------------------------------------------------------------------
// FINAL CHECKS
// -----------------------------------------------------------------------

// Assert that the values are correctly set
let (program_info_res, facts_registry_res) =
tokio::join!(appchain.get_program_info().call(), appchain.get_facts_registry().call());

let (actual_program_hash, actual_config_hash) = program_info_res?;
let facts_registry = facts_registry_res?;

if actual_program_hash != PROGRAM_HASH {
return Err(ContractInitError::InvalidProgramHash {
actual: actual_program_hash,
expected: PROGRAM_HASH,
});
}

if actual_config_hash != config_hash {
return Err(ContractInitError::InvalidConfigHash {
actual: actual_config_hash,
expected: config_hash,
});
}

if facts_registry != ATLANTIC_FACT_REGISTRY_SEPOLIA.into() {
return Err(ContractInitError::InvalidFactRegistry {
actual: facts_registry.into(),
expected: ATLANTIC_FACT_REGISTRY_SEPOLIA,
});
}

Ok(deployed_appchain_contract.into())
}
.await;

match result {
Ok(addr) => sp.success(&format!("Deployment successful ({addr})")),
Err(..) => sp.fail("Deployment failed"),
}
result
}

/// Error that can happen during the initialization of the core contract.
#[derive(Error, Debug)]
pub enum ContractInitError {
#[error("failed to declare contract: {0:#?}")]
DeclarationError(AccountError<<InitializerAccount as Account>::SignError>),

#[error("failed to deploy contract: {0:#?}")]
DeploymentError(AccountError<<InitializerAccount as Account>::SignError>),

#[error("failed to initialize contract: {0:#?}")]
Initialization(AccountError<<InitializerAccount as Account>::SignError>),

#[error(
"invalid program info: program hash mismatch - expected {expected:#x}, got {actual:#x}"
)]
InvalidProgramHash { expected: Felt, actual: Felt },

#[error("invalid program info: config hash mismatch - expected {expected:#x}, got {actual:#x}")]
InvalidConfigHash { expected: Felt, actual: Felt },

#[error("invalid program state: fact registry mismatch - expected {expected:}, got {actual}")]
InvalidFactRegistry { expected: Felt, actual: Felt },

#[error(transparent)]
TxWaitingError(#[from] TransactionWaitingError),

#[error("failed parsing contract class: {0}")]
ContractParsing(#[from] ContractClassFromStrError),

#[error(transparent)]
ContractClassCompilation(#[from] ContractClassCompilationError),

#[error(transparent)]
ComputeClassHash(#[from] ComputeClassHashError),

#[error(transparent)]
Provider(#[from] ProviderError),

#[error(transparent)]
Other(#[from] anyhow::Error),
}

impl From<cairo_serde::Error> for ContractInitError {
fn from(value: cairo_serde::Error) -> Self {
match value {
cairo_serde::Error::Provider(e) => Self::Provider(e),
_ => Self::Other(anyhow!(value)),
}
}
}

fn prepare_contract_declaration_params(
class: ContractClass,
) -> Result<(FlattenedSierraClass, CompiledClassHash)> {
let casm_hash = class.clone().compile()?.class_hash()?;

let rpc_class = RpcContractClass::try_from(class).expect("should be valid");
let RpcContractClass::Class(class) = rpc_class else { unreachable!("unexpected legacy class") };
let flattened: FlattenedSierraClass = class.try_into()?;

Ok((flattened, casm_hash))
}

// NOTE: The reason why we're using the same address for both fee tokens is because we don't yet
// support having native fee token on the chain.
fn compute_config_hash(chain_id: Felt, fee_token: Felt) -> Felt {
compute_starknet_os_config_hash(chain_id, fee_token, fee_token)
}

// /~https://github.com/starkware-libs/cairo-lang/blob/a86e92bfde9c171c0856d7b46580c66e004922f3/src/starkware/starknet/core/os/os_config/os_config.cairo#L1-L39
fn compute_starknet_os_config_hash(
chain_id: Felt,
deprecated_fee_token: Felt,
fee_token: Felt,
) -> Felt {
// A constant representing the StarkNet OS config version.
const STARKNET_OS_CONFIG_VERSION: Felt = short_string!("StarknetOsConfig2");

compute_hash_on_elements(&[
STARKNET_OS_CONFIG_VERSION,
chain_id,
deprecated_fee_token,
fee_token,
])
}

#[cfg(test)]
mod tests {
use katana_primitives::{felt, Felt};
use starknet::core::chain_id::{MAINNET, SEPOLIA};

use super::compute_starknet_os_config_hash;

const ETH_FEE_TOKEN: Felt =
felt!("0x049d36570d4e46f48e99674bd3fcc84644ddd6b96f7c741b1562b82f9e004dc7");
const STRK_FEE_TOKEN: Felt =
felt!("0x04718f5a0fc34cc1af16a1cdee98ffb20c31f5cd61d6ab07201858f4287c938d");

// Source:
//
// - /~https://github.com/starkware-libs/cairo-lang/blob/8e11b8cc65ae1d0959328b1b4a40b92df8b58595/src/starkware/starknet/core/os/os_config/os_config_hash.json#L4
// - https://docs.starknet.io/tools/important-addresses/#fee_tokens
#[rstest::rstest]
#[case::mainnet(felt!("0x5ba2078240f1585f96424c2d1ee48211da3b3f9177bf2b9880b4fc91d59e9a2"), MAINNET)]
#[case::testnet(felt!("0x504fa6e5eb930c0d8329d4a77d98391f2730dab8516600aeaf733a6123432"), SEPOLIA)]
fn calculate_config_hash(#[case] config_hash: Felt, #[case] chain: Felt) {
let computed = compute_starknet_os_config_hash(chain, ETH_FEE_TOKEN, STRK_FEE_TOKEN);
assert_eq!(computed, config_hash);
}
}
Loading

0 comments on commit 57cfe1e

Please sign in to comment.