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

Implement block tree for the domain v2 #1650

Merged
merged 15 commits into from
Jul 18, 2023
Merged

Implement block tree for the domain v2 #1650

merged 15 commits into from
Jul 18, 2023

Conversation

NingLin-P
Copy link
Member

@NingLin-P NingLin-P commented Jul 6, 2023

close #1610

This PR implements block tree for the domain v2, which tracks the execution receipt for all domains.

This PR also has revised the generate_genesis_state_root host function to integrate the domain registry with block tree, but the genesis state root is not unique for every domain instance as the spec required. Because the genesis block builder takes runtime genesis config as an input and which is the same for all domain instances (the default value), adding digest to the genesis block header won't help as it won't affect the state root, we need to handle this issue in #1631.

Code contributor checklist:

@NingLin-P
Copy link
Member Author

CI failed due to a similar issue of #1599, namely the GenesisReceiptExtension is not registered when building the genesis storage, but this time we do not have a similar workaround. cc @liuchengxu

@vedhavyas
Copy link
Member

vedhavyas commented Jul 8, 2023

@NingLin-P Since this is at genesis, can't we schedule it for next block? This would be fastest route though we can also look into upstream and find out reason why extensions are not enabled at genesis. We do have a few TODOs for upstream for the same issue so we have a way to track these TODOs

Copy link
Contributor

@liuchengxu liuchengxu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Some early comments, will take another look later.

Regarding the genesis extension, I have created an issue upstream paritytech/substrate#14541. As Ved said, the quick workaround at present is to defer the genesis receipt generation to sometime later, we can always revisit once it's fixed upstream, and we can contribute to it for sure.

crates/sp-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Show resolved Hide resolved
crates/sp-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/sp-domains/src/lib.rs Outdated Show resolved Hide resolved
Copy link
Member

@vedhavyas vedhavyas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Makes sense overall. Will take another look again.

crates/pallet-domains/src/lib.rs Show resolved Hide resolved
}

fn successful_bundle_hashes() -> Vec<H256> {
Domains::successful_bundles()
Domains::successful_bundles_of_all_domains()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I dont think this is correct. This func is called under the assumption that these belong to a single domain. So should be scoped to specific domain.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not true, this func is called by bundle validator and is expected bundle of all domains, the use case you mentioned use Domains::successful_bundles(domain_id).

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmmm if the consensus node wants to fetch these bundles, which is the only case part that needs to see all domain bundles that I could think of, the result of this function is not very useful IMO.

Copy link
Member Author

@NingLin-P NingLin-P Jul 10, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmmm if the consensus node wants to fetch these bundles, which is the only case part that needs to see all domain bundles that I could think of

This is the exact use case, we use the bundle validator in the consensus node's tx pool to detect duplicated bundles otherwise the attacker can use them to occupy the block space without cost.

crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/sp-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/tests.rs Show resolved Hide resolved
crates/pallet-domains/src/tests.rs Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
…need runtime_genesis_config parameter

Signed-off-by: linning <[email protected]>
The v1 primitives can be removed and replaced by them  after the domain client
retired all of the v1 usage.

Signed-off-by: linning <[email protected]>
This commit also remove the unused ReceiptsPruningDepth

Signed-off-by: linning <[email protected]>
This commit includes the block tree types, storage items, the submit_bundle_v2 call and
pre_dispatch/validate_unsigned verification.

Signed-off-by: linning <[email protected]>
Signed-off-by: linning <[email protected]>
@NingLin-P
Copy link
Member Author

This PR is rebased, the changes of resolving the comments are put into the corresponding commit, PATL, thanks!

crates/pallet-domains/src/lib.rs Show resolved Hide resolved
crates/subspace-runtime/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/benchmarking.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/domain_registry.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
let next_number = HeadDomainNumber::<T>::get(domain_id)
.checked_add(&One::one())
.ok_or::<Error<T>>(BlockTreeError::MaxHeadDomainNumber.into())?;
HeadDomainNumber::<T>::set(domain_id, next_number);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The HeadDomainNumber is stored in the consensus chain, suppose there is a consensus block that contains extrinsic that updates a domain 'sHeadDomainNumber from 1 to 2. Operators who have processed this consensus block (drive & import a domain block) its best block number update from 1 to 2, which match your above statement.

How do some one definitely prove that unless there is an execution receipt to corroborate that statement ?

Copy link
Contributor

@liuchengxu liuchengxu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Mostly make sense, nice work!

crates/pallet-domains/src/block_tree.rs Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
Copy link
Member

@vedhavyas vedhavyas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Left a few questions on few storage items. I'm not sure why we need them and those seems to cause more confusion to me actually

crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
let next_number = HeadDomainNumber::<T>::get(domain_id)
.checked_add(&One::one())
.ok_or::<Error<T>>(BlockTreeError::MaxHeadDomainNumber.into())?;
HeadDomainNumber::<T>::set(domain_id, next_number);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure if you understand my reasoning. Per my understanding, just submitting domain bundle is not the correct step to consider a domain chain has progressed unless there is ER associated with it. Lets take the current DomainBlock, It needs to have ER and per our definition of DomainBlockNumber, if there is a new domain block, then there is new head backing. Right now, we just do it on first bundle, which seem odd TBH.

// consensus block, which also mean a domain block will be produced thus update `HeadDomainNumber`
// to this domain block's block number.
if SuccessfulBundles::<T>::get(domain_id).is_empty() {
let next_number = HeadDomainNumber::<T>::get(domain_id)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

another parallel question, why are we extending the head all receipt types? I'm not sure I see this handled. What if the first bundle is actually creating a new branch?
We should only extend for type NewHead, no ?

Copy link
Member Author

@NingLin-P NingLin-P Jul 13, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

#1650 (comment) may answer this, although bundle and ER are submitted together, they are building different things. ER is used to build the block tree and may create a new branch in the tree while the bundle is used to build the domain chain.

crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
execution_receipt: &ExecutionReceiptOf<T>,
) -> ReceiptType {
let receipt_number = execution_receipt.domain_block_number;
let head_receipt_number = HeadReceiptNumber::<T>::get(domain_id);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I dont understant this. IIUC head_receipt_number should be the head of the fork this receipt is part of. But head receipt number you are using may or may not be the same

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The HeadReceiptNumber is applied to the whole block tree not any specific fork in the tree, refer #1650 (comment).

/// number will be produce. Used as a pointer in `ExecutionInbox` to identify the current under building
/// domain block, also used as a mapping of consensus block number to domain block number.
#[pallet::storage]
pub(super) type HeadDomainNumber<T: Config> =
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not able to find specific need for this storage item and HeadReceiptNumber. Can you help me understand how these are useful ?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

HeadDomainNumber is used to index the ExecutionInbox as the comment explains, for HeadReceiptNumber refer #1650 (comment).

The domain tests are broken becasue the genesis domain is delay for 1 block
to initialize, thus the domain block can not produce block for the slot 1,
which cause the tests stuck. This commit fix it by produce 1 consensus block
upon setup to initialize the genesis domain.

Signed-off-by: linning <[email protected]>
Copy link
Contributor

@liuchengxu liuchengxu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

looks good to me to proceed, a few minor comments, none of them are blockers.

Regarding whether we should take all bundles that are possibly from different domain branches as the input for the next domain block, we can revisit once it's more clearly defined in the spec.

crates/pallet-domains/src/block_tree.rs Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/lib.rs Outdated Show resolved Hide resolved
crates/pallet-domains/src/block_tree.rs Show resolved Hide resolved
Copy link
Member

@vedhavyas vedhavyas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Though I'm still skeptical about the assumptions made specifically on bundles, I think we can revisit them once we have some concrete examples that might warrant for a refactor.

Please fix conflicts and we can get this in 👍🏼
Thank you for being patient @NingLin-P

crates/pallet-domains/src/block_tree.rs Show resolved Hide resolved
@NingLin-P NingLin-P enabled auto-merge July 18, 2023 17:13
@NingLin-P NingLin-P merged commit d121d12 into main Jul 18, 2023
@NingLin-P NingLin-P deleted the block-tree-pr branch July 18, 2023 17:52
ParthDesai added a commit to autonomys/subspace-pulsar-sdk that referenced this pull request Aug 4, 2023
ParthDesai added a commit to autonomys/subspace-pulsar-sdk that referenced this pull request Aug 15, 2023
ParthDesai added a commit to autonomys/subspace-pulsar-sdk that referenced this pull request Aug 15, 2023
ParthDesai added a commit to autonomys/subspace-pulsar-sdk that referenced this pull request Aug 15, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Block tree
3 participants