Skip to content
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

versioning #1637

Merged
merged 52 commits into from
Jul 10, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
52 commits
Select commit Hold shift + click to select a range
0555b0b
Replace public header fields with getters
jbearer May 23, 2024
f605483
WIP upgradable types
jbearer May 28, 2024
8adbdfd
v2
imabdulbasit Jun 24, 2024
365d884
merge origin/main
imabdulbasit Jun 24, 2024
0f225bb
Merge branch 'main' into ab/ver
imabdulbasit Jun 24, 2024
9896587
cargo sort
imabdulbasit Jun 24, 2024
6e2e9e6
block impls
imabdulbasit Jun 25, 2024
ad046c3
remove unused imports
imabdulbasit Jun 25, 2024
278f1fb
export types
imabdulbasit Jun 25, 2024
82ec12f
group imports
imabdulbasit Jun 25, 2024
bcff34a
v3 header
imabdulbasit Jun 25, 2024
eac798d
lint
imabdulbasit Jun 25, 2024
69fce8b
remove v2 header definition
imabdulbasit Jun 25, 2024
8819268
move crates to workspace
imabdulbasit Jun 25, 2024
4dae934
remove types from sequencer crate
imabdulbasit Jun 26, 2024
26876bc
group imports
imabdulbasit Jun 26, 2024
aaa6f11
serialization
imabdulbasit Jul 1, 2024
0fc4398
fix test
imabdulbasit Jul 1, 2024
df2b9b5
fix deserialization
imabdulbasit Jul 2, 2024
0dbc44b
Merge remote-tracking branch 'origin/main' into ab/ver
imabdulbasit Jul 2, 2024
43d9331
cargo sort
imabdulbasit Jul 2, 2024
d68d146
lint
imabdulbasit Jul 2, 2024
013a32c
fix payload::empty()
imabdulbasit Jul 2, 2024
2c28e6c
fix: visit_map deserialization
imabdulbasit Jul 2, 2024
daeb6c4
use serde_json map
imabdulbasit Jul 2, 2024
4815708
cargo sort
imabdulbasit Jul 2, 2024
7d197dd
Merge branch 'main' into ab/ver
imabdulbasit Jul 2, 2024
420af73
fix ResolvableChainConfigOrVersion deserialization
imabdulbasit Jul 2, 2024
41d0bd9
Merge remote-tracking branch 'origin/main' into ab/ver
imabdulbasit Jul 3, 2024
179d80a
move out traits and errors
imabdulbasit Jul 3, 2024
639a903
fix imports
imabdulbasit Jul 3, 2024
7fa7f20
move errors into impls
imabdulbasit Jul 8, 2024
d0ac37b
Merge remote-tracking branch 'origin/main' into ab/ver
imabdulbasit Jul 8, 2024
1dc9da5
remove ResolvableChainConfig and generic parameters from EitherOrVersion
imabdulbasit Jul 9, 2024
18531d2
remove committable trait implementation for v1 Header
imabdulbasit Jul 9, 2024
faf5f1f
add comments for VersionedHeader
imabdulbasit Jul 9, 2024
7f3b541
remove StructFields enum
imabdulbasit Jul 10, 2024
02c9c0e
add reference tests for v1, v2, and v3 header
imabdulbasit Jul 10, 2024
95b06ab
move reference_tests to types crate
imabdulbasit Jul 10, 2024
c7d93df
fix: v2 and v3 reference test
imabdulbasit Jul 10, 2024
8c8b193
Merge remote-tracking branch 'origin/main' into ab/ver
imabdulbasit Jul 10, 2024
e7a2926
fix comments
imabdulbasit Jul 10, 2024
5a5a4da
add comments for Header::genesis()
imabdulbasit Jul 10, 2024
bfbf42f
add comments for Header::create panic
imabdulbasit Jul 10, 2024
2676df2
add comment for global error issue
imabdulbasit Jul 10, 2024
27e928f
add comment for global error issue
imabdulbasit Jul 10, 2024
1127c84
build genesis header for the current sequencer version
imabdulbasit Jul 10, 2024
bfa3d9c
version sub-directories for data dir
imabdulbasit Jul 10, 2024
4aad7ba
move messages.json and messages.bin to data dir
imabdulbasit Jul 10, 2024
be85894
update data readme
imabdulbasit Jul 10, 2024
a6cec5e
cargo update
imabdulbasit Jul 10, 2024
d2b7b51
include sub dirs binaries in gitignore
imabdulbasit Jul 10, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
154 changes: 150 additions & 4 deletions types/src/v0/header.rs
Original file line number Diff line number Diff line change
@@ -1,20 +1,166 @@
use std::fmt;

use committable::Commitment;
use hotshot_query_service::VidCommitment;
use hotshot_types::utils::BuilderCommitment;
use serde::{Deserialize, Serialize};
use itertools::Either;
use serde::de::{self, MapAccess, SeqAccess, Visitor};
use serde::{Deserialize, Deserializer, Serialize, Serializer};
use vbs::version::Version;

use crate::{
v0_1, v0_2, v0_3, BlockMerkleCommitment, BuilderSignature, FeeInfo, FeeMerkleCommitment,
L1BlockInfo, NsTable, ResolvableChainConfig,
v0_1, v0_2, v0_3, BlockMerkleCommitment, BuilderSignature, ChainConfig, FeeInfo,
FeeMerkleCommitment, L1BlockInfo, NsTable, ResolvableChainConfig,
};

#[derive(Clone, Debug, Deserialize, Serialize, Hash, PartialEq, Eq)]
#[derive(Clone, Debug, Hash, PartialEq, Eq)]
pub enum Header {
V1(v0_1::Header),
V2(v0_2::Header),
V3(v0_3::Header),
}

#[derive(Clone, Debug, Deserialize, Serialize)]
enum EitherOrVersion<ChainConfig, Commitment> {
Left(ChainConfig),
Right(Commitment),
Version(Version),
}

#[derive(Clone, Debug, Deserialize, Serialize)]
pub struct ResolvableChainConfigOrVersion {
chain_config: EitherOrVersion<ChainConfig, Commitment<ChainConfig>>,
}

impl From<Version> for ResolvableChainConfigOrVersion {
fn from(version: Version) -> Self {
Self {
chain_config: EitherOrVersion::Version(version),
}
}
}

impl From<ResolvableChainConfig> for ResolvableChainConfigOrVersion {
fn from(v: ResolvableChainConfig) -> Self {
let value = match v.chain_config {
Either::Left(cfg) => EitherOrVersion::Left(cfg),
Either::Right(commit) => EitherOrVersion::Right(commit),
};

Self {
chain_config: value,
}
}
}

impl Serialize for Header {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
match self {
Self::V1(header) => header.serialize(serializer),
Self::V2(fields) => crate::v0_2::VersionedHeader {
version: Version { major: 0, minor: 2 }.into(),
fields: fields.clone(),
}
.serialize(serializer),
Self::V3(fields) => crate::v0_3::VersionedHeader {
version: Version { major: 0, minor: 3 }.into(),
fields: fields.clone(),
}
.serialize(serializer),
}
}
}

impl<'de> Deserialize<'de> for Header {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
struct HeaderVisitor;

#[derive(Deserialize)]
#[serde(field_identifier, rename_all = "lowercase")]
enum StructFields {
Version,
Fields,
}
jbearer marked this conversation as resolved.
Show resolved Hide resolved

impl<'de> Visitor<'de> for HeaderVisitor {
type Value = Header;

fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("Header")
}

fn visit_seq<V>(self, mut seq: V) -> Result<Self::Value, V::Error>
where
V: SeqAccess<'de>,
{
let chain_config_or_version: ResolvableChainConfigOrVersion =
seq.next_element()?.unwrap();
imabdulbasit marked this conversation as resolved.
Show resolved Hide resolved
match chain_config_or_version.chain_config {
// For v0.1, the first field in the sequence of fields is the first field of the struct, so we call a function to get the rest of
// the fields from the sequence and pack them into the struct.
EitherOrVersion::Left(cfg) => Ok(Header::V1(
v0_1::Header::deserialize_with_chain_config(cfg.into(), seq)?,
)),
EitherOrVersion::Right(commit) => Ok(Header::V1(
v0_1::Header::deserialize_with_chain_config(commit.into(), seq)?,
)),
// For all versions, the first "field" is not actually part of the `Header` struct (you can think of it as the first field of the virtual
jbearer marked this conversation as resolved.
Show resolved Hide resolved
// `VersionedHeader` struct, with the `Header` being the second field. We just delegate directly to the derived deserialization
// impl for the appropriate version.
EitherOrVersion::Version(Version { major: 0, minor: 2 }) => Ok(Header::V2(
seq.next_element()?
.ok_or_else(|| de::Error::missing_field("fields"))?,
)),
EitherOrVersion::Version(Version { major: 0, minor: 3 }) => Ok(Header::V3(
seq.next_element()?
.ok_or_else(|| de::Error::missing_field("fields"))?,
)),
_ => Err(serde::de::Error::custom("invalid version")),
}
}

fn visit_map<V>(self, mut map: V) -> Result<Header, V::Error>
where
V: MapAccess<'de>,
{
// retrieving the first entry from the map, which could either be a
// `chain_config` (for v0.1 headers) or a `version` (for `VersionedHeader` structs).
let Some((_, chain_config_or_version)) =
jbearer marked this conversation as resolved.
Show resolved Hide resolved
map.next_entry::<String, ResolvableChainConfigOrVersion>()?
else {
return Err(de::Error::missing_field("version"));
};

// Match to determine the appropriate header version.
match chain_config_or_version.chain_config {
EitherOrVersion::Left(cfg) => Ok(Header::V1(
v0_1::Header::deserialize_with_chain_config_map(cfg.into(), map)?,
)),
EitherOrVersion::Right(commit) => Ok(Header::V1(
v0_1::Header::deserialize_with_chain_config_map(commit.into(), map)?,
)),
EitherOrVersion::Version(Version { major: 0, minor: 2 }) => {
Ok(Header::V2(map.next_entry::<String, _>()?.unwrap().1))
}
EitherOrVersion::Version(Version { major: 0, minor: 3 }) => {
Ok(Header::V3(map.next_entry::<String, _>()?.unwrap().1))
}
_ => Err(serde::de::Error::custom("invalid version")),
}
}
}

const FIELDS: &[&str] = &["version", "fields"];
deserializer.deserialize_struct("Header", FIELDS, HeaderVisitor)
}
}

impl Header {
#[allow(clippy::too_many_arguments)]
pub fn create(
Expand Down
Loading