Skip to content

Commit

Permalink
Refactor CelestiaAddress (#551)
Browse files Browse the repository at this point in the history
* Refactor CelestiaAddress and its internal representation
  • Loading branch information
citizen-stig authored and neysofu committed Aug 21, 2023
1 parent 6e368cb commit d6c67d3
Show file tree
Hide file tree
Showing 10 changed files with 194 additions and 28 deletions.
183 changes: 166 additions & 17 deletions adapters/celestia/src/verifier/address.rs
Original file line number Diff line number Diff line change
@@ -1,56 +1,91 @@
use std::fmt::{Display, Formatter};
use std::str::FromStr;

use bech32::WriteBase32;
use borsh::{BorshDeserialize, BorshSerialize};
use serde::{Deserialize, Serialize};
use sov_rollup_interface::AddressTrait;
use thiserror::Error;

/// Human Readable Part: "celestia" for Celestia network
const HRP: &str = "celestia";
/// Bech32 variant is used for Celestia and CosmosSDK
const VARIANT: bech32::Variant = bech32::Variant::Bech32;

/// Representation of the address in the Celestia network
/// https://github.com/celestiaorg/celestia-specs/blob/e59efd63a2165866584833e91e1cb8a6ed8c8203/src/specs/data_structures.md#address
/// Spec says: "Addresses have a length of 32 bytes.", but in reality it is 32 `u5` elements, which can be compressed as 20 bytes.
/// TODO: Switch to bech32::u5 when it has repr transparent: https://github.com/Sovereign-Labs/sovereign-sdk/issues/646
#[derive(
Debug, PartialEq, Clone, Eq, Serialize, Deserialize, BorshDeserialize, BorshSerialize, Hash,
)]
// Raw ASCII bytes, including HRP
// TODO: https://github.com/Sovereign-Labs/sovereign-sdk/issues/469
pub struct CelestiaAddress(Vec<u8>);
pub struct CelestiaAddress([u8; 32]);

impl AsRef<[u8]> for CelestiaAddress {
fn as_ref(&self) -> &[u8] {
self.0.as_ref()
}
}

/// Decodes slice of bytes into CelestiaAddress
/// Treats it as string if it starts with HRP and the rest is valid ASCII
/// Otherwise just checks if it contains valid `u5` elements and has the correct length.
impl<'a> TryFrom<&'a [u8]> for CelestiaAddress {
type Error = anyhow::Error;

fn try_from(value: &'a [u8]) -> Result<Self, Self::Error> {
Ok(Self(value.to_vec()))
if value.starts_with(HRP.as_bytes()) && value.is_ascii() {
// safety, because we checked that it is ASCII
let s = unsafe { std::str::from_utf8_unchecked(value) };
return CelestiaAddress::from_str(s).map_err(|e| anyhow::anyhow!("{}", e));
}
if value.len() != 32 {
anyhow::bail!("An address must be 32 u5 long");
}
let mut raw_address = [0u8; 32];
for (idx, &item) in value.iter().enumerate() {
bech32::u5::try_from_u8(item)
.map_err(|e| anyhow::anyhow!("Element at {} is not u5: {}", idx, e))?;
raw_address[idx] = item;
}
Ok(Self(raw_address))
}
}

/// Panics if any element is not in range 0..32 (u5)
/// TODO: Will be removed after https://github.com/Sovereign-Labs/sovereign-sdk/issues/493
impl From<[u8; 32]> for CelestiaAddress {
fn from(value: [u8; 32]) -> Self {
// TODO: This is completely broken with current implementation.
// https://github.com/Sovereign-Labs/sovereign-sdk/issues/469
Self(value.to_vec())
for item in value {
bech32::u5::try_from_u8(item).unwrap();
}
Self(value)
}
}

impl Display for CelestiaAddress {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
let ascii_string = String::from_utf8_lossy(&self.0);
write!(f, "{}", ascii_string)
let mut w = bech32::Bech32Writer::new(HRP, VARIANT, f)?;
for elem in self.0.iter() {
// It is ok to unwrap, because we always sanitize data
w.write_u5(bech32::u5::try_from_u8(*elem).unwrap())?;
}
w.finalize()
}
}

#[derive(Clone, Debug, Error)]
#[derive(Clone, Debug, Error, PartialEq)]
/// An error which occurs while decoding a `CelestialAddress` from a string.
pub enum CelestiaAddressFromStrError {
/// The address has an invalid human readable prefix. Valid addresses must start with the prefix 'celestia'.
#[error("The address has an invalid human readable prefix. Valid addresses must start with the prefix 'celestia', but this one began with {0}")]
/// The address has an invalid human-readable prefix.
/// Valid addresses must start with the prefix 'celestia'.
#[error("The address has an invalid human-readable prefix. Valid addresses must start with the prefix 'celestia', but this one began with {0}")]
InvalidHumanReadablePrefix(String),
/// The address could note be decoded as valid bech32
/// The address has an invalid human-readable prefix.
/// Valid addresses must start with the prefix 'celestia'.
#[error("The address has an invalid bech32 variant. Valid addresses must be encoded in Bech32, but this is encoded in Bech32m")]
InvalidVariant,
/// The address could not be decoded as valid bech32
#[error("The address could not be decoded as valid bech32: {0}")]
InvalidBech32(#[from] bech32::Error),
}
Expand All @@ -59,12 +94,24 @@ impl FromStr for CelestiaAddress {
type Err = CelestiaAddressFromStrError;

fn from_str(s: &str) -> Result<Self, Self::Err> {
// This could be the way to save memory:
let (hrp, _raw_address_u5, _variant) = bech32::decode(s)?;
let (hrp, raw_address_u5, variant) = bech32::decode(s)?;
if hrp != HRP {
return Err(CelestiaAddressFromStrError::InvalidHumanReadablePrefix(hrp));
}
let value = s.as_bytes().to_vec();
if variant != VARIANT {
return Err(CelestiaAddressFromStrError::InvalidVariant);
}
if raw_address_u5.len() != 32 {
return Err(CelestiaAddressFromStrError::InvalidBech32(
bech32::Error::InvalidLength,
));
}

let mut value: [u8; 32] = [0; 32];

for (idx, &item) in raw_address_u5.iter().enumerate() {
value[idx] = item.to_u8();
}
Ok(Self(value))
}
}
Expand All @@ -73,11 +120,16 @@ impl AddressTrait for CelestiaAddress {}

#[cfg(test)]
mod tests {
use std::hint::black_box;

use bech32::ToBase32;
use proptest::prelude::*;

use super::*;

#[test]
fn test_address_display_from_string() {
let raw_address_str = "celestia1w7wcupk5gswj25c0khnkey5fwmlndx6t5aarmk";
let raw_address_str = "celestia1hvp2nfz3r6nqt8mlrzqf9ctwle942tkr0wql75";
let address = CelestiaAddress::from_str(raw_address_str).unwrap();
let output = format!("{}", address);
assert_eq!(raw_address_str, output);
Expand All @@ -91,4 +143,101 @@ mod tests {
let output = format!("{}", address);
assert_eq!(raw_address_str, output);
}

#[test]
fn test_from_str_and_from_slice_same() {
let raw_address_str = "celestia1w7wcupk5gswj25c0khnkey5fwmlndx6t5aarmk";
let raw_address_array = *b"celestia1w7wcupk5gswj25c0khnkey5fwmlndx6t5aarmk";

let address_from_str = CelestiaAddress::from_str(raw_address_str).unwrap();
let address_from_slice = CelestiaAddress::try_from(&raw_address_array[..]).unwrap();

assert_eq!(address_from_str, address_from_slice);
}

// 20 u8 -> 32 u5
fn check_from_bytes_as_ascii(input: [u8; 20]) {
let encoded = bech32::encode("celestia", input.to_base32(), VARIANT).unwrap();
let bytes = encoded.as_bytes();
let address = CelestiaAddress::try_from(bytes);
assert!(address.is_ok());
let address = address.unwrap();
let output = format!("{}", address);
assert_eq!(encoded, output);
}

// 20 u8 -> 32 u5
fn check_from_as_ref(input: [u8; 20]) {
let encoded = bech32::encode("celestia", input.to_base32(), VARIANT).unwrap();
let address1 = CelestiaAddress::from_str(&encoded).unwrap();
let bytes = address1.as_ref();
let address = CelestiaAddress::try_from(bytes);
assert!(address.is_ok());
let address = address.unwrap();
let output = format!("{}", address);
assert_eq!(encoded, output);
}

// 20 u8 -> 32 u5
fn check_borsh(input: [u8; 20]) {
let address_str = bech32::encode("celestia", input.to_base32(), VARIANT).unwrap();

let address = CelestiaAddress::from_str(&address_str).unwrap();
let serialized = BorshSerialize::try_to_vec(&address).unwrap();
let deserialized = CelestiaAddress::try_from_slice(&serialized).unwrap();

assert_eq!(deserialized, address);

let address_str2 = format!("{}", deserialized);
assert_eq!(address_str2, address_str);
}

proptest! {
#[test]
fn test_try_from_any_slice(input in prop::collection::vec(any::<u8>(), 0..100)) {
let _ = black_box(CelestiaAddress::try_from(&input[..]));
}

#[test]
fn test_from_str_anything(input in "\\PC*") {
let _ = black_box(CelestiaAddress::from_str(&input));
}

#[test]
// According to spec, alphanumeric characters excluding "1" "b" "i" and "o"
fn test_from_str_lowercase_ascii(input in "celestia1[023456789ac-hj-np-z]{38}") {
let result = CelestiaAddress::from_str(&input);
match result {
Ok(address) => {
let output = format!("{}", address);
assert_eq!(input, output);
}
Err(err) => {
assert_eq!(CelestiaAddressFromStrError::InvalidBech32(bech32::Error::InvalidChecksum), err);
},
}
}

#[test]
fn test_try_from_ascii_slice(input in proptest::array::uniform20(0u8..=255)) {
check_from_bytes_as_ascii(input);
}

#[test]
fn test_try_as_ref_from(input in proptest::array::uniform20(0u8..=255)) {
check_from_as_ref(input);
}

#[test]
fn test_borsh(input in proptest::array::uniform20(0u8..=255)) {
check_borsh(input);
}

#[test]
fn test_try_from_array(arr in proptest::array::uniform32(0u8..32)) {
let address = CelestiaAddress::from(arr);
let output = format!("{}", address);
prop_assert!(output.starts_with("celestia"));
}
}
}
2 changes: 1 addition & 1 deletion examples/const-rollup-config/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,4 +5,4 @@ pub const ROLLUP_NAMESPACE_RAW: [u8; 8] = [115, 111, 118, 45, 116, 101, 115, 116

/// The DA address of the sequencer (for now we use a centralized sequencer) in the tests.
/// Here this is the address of the sequencer on the celestia blockchain.
pub const SEQUENCER_DA_ADDRESS: [u8; 47] = *b"celestia1w7wcupk5gswj25c0khnkey5fwmlndx6t5aarmk";
pub const SEQUENCER_DA_ADDRESS: &str = "celestia1w7wcupk5gswj25c0khnkey5fwmlndx6t5aarmk";
3 changes: 1 addition & 2 deletions examples/demo-prover/Cargo.lock

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

5 changes: 4 additions & 1 deletion examples/demo-prover/host/src/main.rs
Original file line number Diff line number Diff line change
@@ -1,11 +1,13 @@
use std::env;
use std::str::FromStr;

use anyhow::Context;
use const_rollup_config::{ROLLUP_NAMESPACE_RAW, SEQUENCER_DA_ADDRESS};
use demo_stf::app::{App, DefaultPrivateKey};
use demo_stf::genesis_config::create_demo_genesis_config;
use jupiter::da_service::{CelestiaService, DaServiceConfig};
use jupiter::types::NamespaceId;
use jupiter::verifier::address::CelestiaAddress;
use jupiter::verifier::{ChainValidityCondition, RollupParams};
use methods::{ROLLUP_ELF, ROLLUP_ID};
use risc0_adapter::host::{Risc0Host, Risc0Verifier};
Expand Down Expand Up @@ -60,10 +62,11 @@ async fn main() -> Result<(), anyhow::Error> {
let is_storage_empty = app.get_storage().is_empty();

if is_storage_empty {
let sequencer_da_address = CelestiaAddress::from_str(SEQUENCER_DA_ADDRESS).unwrap();
let genesis_config = create_demo_genesis_config(
100000000,
sequencer_private_key.default_address(),
SEQUENCER_DA_ADDRESS.to_vec(),
sequencer_da_address.as_ref().to_vec(),
&sequencer_private_key,
&sequencer_private_key,
);
Expand Down
7 changes: 6 additions & 1 deletion examples/demo-prover/methods/guest/src/bin/rollup.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,13 @@

#![no_main]

use std::str::FromStr;

use const_rollup_config::{ROLLUP_NAMESPACE_RAW, SEQUENCER_DA_ADDRESS};
use demo_stf::app::create_zk_app_template;
use demo_stf::ArrayWitness;
use jupiter::types::NamespaceId;
use jupiter::verifier::address::CelestiaAddress;
use jupiter::verifier::{CelestiaSpec, CelestiaVerifier, ChainValidityCondition};
use jupiter::{BlobWithSender, CelestiaHeader};
use risc0_adapter::guest::Risc0Guest;
Expand Down Expand Up @@ -67,11 +70,13 @@ pub fn main() {
.expect("Transaction list must be correct");
env::write(&"Relevant txs verified\n");

// TODO: https://github.com/Sovereign-Labs/sovereign-sdk/issues/647
let rewarded_address = CelestiaAddress::from_str(SEQUENCER_DA_ADDRESS).unwrap();
let output = StateTransition {
initial_state_root: prev_state_root_hash,
final_state_root: result.state_root.0,
validity_condition,
rewarded_address: SEQUENCER_DA_ADDRESS.to_vec(),
rewarded_address: rewarded_address.as_ref().to_vec(),
slot_hash: header.hash(),
};
env::commit(&output);
Expand Down
4 changes: 3 additions & 1 deletion examples/demo-rollup/benches/rollup_bench.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use std::env;
use std::path::PathBuf;
use std::str::FromStr;
use std::sync::Arc;
use std::time::Duration;

Expand Down Expand Up @@ -50,10 +51,11 @@ fn rollup_bench(_bench: &mut Criterion) {

let mut demo = demo_runner.stf;
let sequencer_private_key = DefaultPrivateKey::generate();
let sequencer_da_address = CelestiaAddress::from_str(SEQUENCER_DA_ADDRESS).unwrap();
let demo_genesis_config = create_demo_genesis_config(
100000000,
sequencer_private_key.default_address(),
SEQUENCER_DA_ADDRESS.to_vec(),
sequencer_da_address.as_ref().to_vec(),
&sequencer_private_key,
&sequencer_private_key,
);
Expand Down
4 changes: 3 additions & 1 deletion examples/demo-rollup/benches/rollup_coarse_measure.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use std::env;
use std::path::PathBuf;
use std::str::FromStr;
use std::sync::Arc;
use std::time::{Duration, Instant};

Expand Down Expand Up @@ -98,10 +99,11 @@ async fn main() -> Result<(), anyhow::Error> {

let mut demo = demo_runner.stf;
let sequencer_private_key = DefaultPrivateKey::generate();
let sequencer_da_address = CelestiaAddress::from_str(SEQUENCER_DA_ADDRESS).unwrap();
let demo_genesis_config = create_demo_genesis_config(
100000000,
sequencer_private_key.default_address(),
SEQUENCER_DA_ADDRESS.to_vec(),
sequencer_da_address.as_ref().to_vec(),
&sequencer_private_key,
&sequencer_private_key,
);
Expand Down
5 changes: 4 additions & 1 deletion examples/demo-rollup/src/main.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
use std::env;
use std::str::FromStr;
use std::sync::Arc;

use anyhow::Context;
Expand All @@ -11,6 +12,7 @@ use jupiter::da_service::CelestiaService;
#[cfg(feature = "experimental")]
use jupiter::da_service::DaServiceConfig;
use jupiter::types::NamespaceId;
use jupiter::verifier::address::CelestiaAddress;
use jupiter::verifier::{ChainValidityCondition, RollupParams};
use risc0_adapter::host::Risc0Verifier;
use sov_db::ledger_db::LedgerDB;
Expand Down Expand Up @@ -68,10 +70,11 @@ pub fn get_genesis_config() -> GenesisConfig<DefaultContext> {
hex_key.address,
"Inconsistent key data",
);
let sequencer_da_address = CelestiaAddress::from_str(SEQUENCER_DA_ADDRESS).unwrap();
create_demo_genesis_config(
100000000,
sequencer_private_key.default_address(),
SEQUENCER_DA_ADDRESS.to_vec(),
sequencer_da_address.as_ref().to_vec(),
&sequencer_private_key,
&sequencer_private_key,
)
Expand Down
Loading

0 comments on commit d6c67d3

Please sign in to comment.