Skip to content

Commit

Permalink
Merge pull request #5607 from stacks-network/chore/3.1.0.0.2_changelog
Browse files Browse the repository at this point in the history
Merge master to develop (3.1.0.0.2) via intermediary branch
  • Loading branch information
wileyj authored Jan 8, 2025
2 parents 4706d74 + 015221b commit 464b796
Show file tree
Hide file tree
Showing 7 changed files with 105 additions and 16 deletions.
12 changes: 3 additions & 9 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,15 +16,7 @@ and this project adheres to the versioning scheme outlined in the [README.md](RE
- When a transaction is dropped due to replace-by-fee, the `/drop_mempool_tx` event observer payload now includes `new_txid`, which is the transaction that replaced this dropped transaction. When a transaction is dropped for other reasons, `new_txid` is `null`. [#5381](https://github.com/stacks-network/stacks-core/pull/5381)
- Nodes will assume that all PoX anchor blocks exist by default, and stall initial block download indefinitely to await their arrival (#5502)

## [3.1.0.0.1]

### Added

- A miner will now generate a tenure-extend when at least 70% of the signers have confirmed that they are willing to allow one, via the new timestamp included in block responses. This allows the miner to refresh its budget in between Bitcoin blocks. ([#5476](https://github.com/stacks-network/stacks-core/discussions/5476))

### Changed

## [3.1.0.0.0]
## [3.1.0.0.2]

### Added

Expand All @@ -33,6 +25,8 @@ and this project adheres to the versioning scheme outlined in the [README.md](RE
- `/v2/clarity/marf/:marf_key_hash`
- `/v2/clarity/metadata/:principal/:contract_name/:clarity_metadata_key`
- When a proposed block is validated by a node, the block can be validated even when the block version is different than the node's default ([#5539](https://github.com/stacks-network/stacks-core/pull/5539))
- A miner will now generate a tenure-extend when at least 70% of the signers have confirmed that they are willing to allow one, via the new timestamp included in block responses. This allows the miner to refresh its budget in between Bitcoin blocks. ([#5476](https://github.com/stacks-network/stacks-core/discussions/5476))
- Set the epoch to 3.1 in the Clarity DB upon activation.

### Changed

Expand Down
15 changes: 13 additions & 2 deletions stacks-signer/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,15 +15,26 @@ and this project adheres to the versioning scheme outlined in the [README.md](RE
- Improvements to the stale signer cleanup logic: deletes the prior signer if it has no remaining unprocessed blocks in its database
- Signers now listen to new block events from the stacks node to determine whether a block has been successfully appended to the chain tip

## [3.1.0.0.1.0]
# [3.1.0.0.2.1]

### Added
## Added

## Changed

- Prevent old reward cycle signers from processing block validation response messages that do not apply to blocks from their cycle.

## [3.1.0.0.2.0]

## Added

- **SIP-029 consensus rules, activating in epoch 3.1 at block 875,000** (see [SIP-029](https://github.com/will-corcoran/sips/blob/feat/sip-029-halving-alignment/sips/sip-029/sip-029-halving-alignment.md) for details)

### Changed

- Added tenure extend timestamp to signer block responses
- Added tenure_idle_timeout_secs configuration option for determining when a time-based tenure extend will be accepted


## [3.1.0.0.0.0]

### Added
Expand Down
16 changes: 16 additions & 0 deletions stacks-signer/src/v0/signer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -659,6 +659,14 @@ impl Signer {
// For mutability reasons, we need to take the block_info out of the map and add it back after processing
let mut block_info = match self.signer_db.block_lookup(&signer_signature_hash) {
Ok(Some(block_info)) => {
if block_info.reward_cycle != self.reward_cycle {
// We are not signing for this reward cycle. Ignore the block.
debug!(
"{self}: Received a block validation response for a different reward cycle. Ignore it.";
"requested_reward_cycle" => block_info.reward_cycle,
);
return None;
}
if block_info.is_locally_finalized() {
debug!("{self}: Received block validation for a block that is already marked as {}. Ignoring...", block_info.state);
return None;
Expand Down Expand Up @@ -744,6 +752,14 @@ impl Signer {
}
let mut block_info = match self.signer_db.block_lookup(&signer_signature_hash) {
Ok(Some(block_info)) => {
if block_info.reward_cycle != self.reward_cycle {
// We are not signing for this reward cycle. Ignore the block.
debug!(
"{self}: Received a block validation response for a different reward cycle. Ignore it.";
"requested_reward_cycle" => block_info.reward_cycle,
);
return None;
}
if block_info.is_locally_finalized() {
debug!("{self}: Received block validation for a block that is already marked as {}. Ignoring...", block_info.state);
return None;
Expand Down
6 changes: 5 additions & 1 deletion stackslib/src/chainstate/nakamoto/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3952,7 +3952,11 @@ impl NakamotoChainState {

// is this stacks block the first of a new epoch?
let (applied_epoch_transition, mut tx_receipts) =
StacksChainState::process_epoch_transition(&mut clarity_tx, burn_header_height)?;
StacksChainState::process_epoch_transition(
&mut clarity_tx,
sortition_dbconn.as_burn_state_db(),
burn_header_height,
)?;

debug!(
"Setup block: Processed epoch transition";
Expand Down
26 changes: 23 additions & 3 deletions stackslib/src/chainstate/stacks/db/blocks.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4046,6 +4046,7 @@ impl StacksChainState {
/// Return (applied?, receipts)
pub fn process_epoch_transition(
clarity_tx: &mut ClarityTx,
burn_dbconn: &dyn BurnStateDB,
chain_tip_burn_header_height: u32,
) -> Result<(bool, Vec<StacksTransactionReceipt>), Error> {
// is this stacks block the first of a new epoch?
Expand Down Expand Up @@ -4106,14 +4107,29 @@ impl StacksChainState {
current_epoch = StacksEpochId::Epoch30;
}
StacksEpochId::Epoch30 => {
// no special initialization is needed, since only the coinbase emission
// schedule is changing.
receipts.append(&mut clarity_tx.block.initialize_epoch_3_1()?);
current_epoch = StacksEpochId::Epoch31;
}
StacksEpochId::Epoch31 => {
panic!("No defined transition from Epoch31 forward")
}
}

if current_epoch > StacksEpochId::Epoch2_05 {
// clarity tx should now have the current epoch
assert_eq!(
clarity_tx.block.get_epoch(),
current_epoch,
"FATAL: clarity_tx does not have the current epoch"
);

// clarity DB should now have the current epoch
assert_eq!(
clarity_tx.block.get_clarity_db_epoch_version(burn_dbconn)?,
current_epoch,
"FATAL: clarity DB does not report the current epoch"
);
}
}
}

Expand Down Expand Up @@ -5200,7 +5216,11 @@ impl StacksChainState {

// is this stacks block the first of a new epoch?
let (applied_epoch_transition, mut tx_receipts) =
StacksChainState::process_epoch_transition(&mut clarity_tx, burn_tip_height)?;
StacksChainState::process_epoch_transition(
&mut clarity_tx,
burn_dbconn,
burn_tip_height,
)?;

debug!(
"Setup block: Processed epoch transition at {}/{}",
Expand Down
41 changes: 41 additions & 0 deletions stackslib/src/clarity_vm/clarity.rs
Original file line number Diff line number Diff line change
Expand Up @@ -226,6 +226,21 @@ impl ClarityBlockConnection<'_, '_> {
None => None,
}
}

/// Load the epoch ID from the clarity DB.
/// Used to sanity-check epoch transitions.
pub fn get_clarity_db_epoch_version(
&mut self,
burn_state_db: &dyn BurnStateDB,
) -> Result<StacksEpochId, Error> {
let mut db = self.datastore.as_clarity_db(self.header_db, burn_state_db);
// NOTE: the begin/roll_back shouldn't be necessary with how this gets used in practice,
// but is put here defensively.
db.begin();
let result = db.get_clarity_epoch_version();
db.roll_back()?;
Ok(result?)
}
}

impl ClarityInstance {
Expand Down Expand Up @@ -1524,6 +1539,32 @@ impl<'a> ClarityBlockConnection<'a, '_> {
})
}

pub fn initialize_epoch_3_1(&mut self) -> Result<Vec<StacksTransactionReceipt>, Error> {
// use the `using!` statement to ensure that the old cost_tracker is placed
// back in all branches after initialization
using!(self.cost_track, "cost tracker", |old_cost_tracker| {
// epoch initialization is *free*.
// NOTE: this also means that cost functions won't be evaluated.
self.cost_track.replace(LimitedCostTracker::new_free());
self.epoch = StacksEpochId::Epoch31;
self.as_transaction(|tx_conn| {
// bump the epoch in the Clarity DB
tx_conn
.with_clarity_db(|db| {
db.set_clarity_epoch_version(StacksEpochId::Epoch31)?;
Ok(())
})
.unwrap();

// require 3.1 rules henceforth in this connection as well
tx_conn.epoch = StacksEpochId::Epoch31;
});

debug!("Epoch 3.1 initialized");
(old_cost_tracker, Ok(vec![]))
})
}

pub fn start_transaction_processing<'c>(&'c mut self) -> ClarityTransactionConnection<'c, 'a> {
let store = &mut self.datastore;
let cost_track = &mut self.cost_track;
Expand Down
5 changes: 4 additions & 1 deletion stackslib/src/core/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -496,7 +496,10 @@ pub static STACKS_EPOCH_3_0_MARKER: u8 = 0x0b;

/// Stacks 3.1 epoch marker. All block-commits in 3.1 must have a memo bitfield with this value
/// *or greater*.
pub static STACKS_EPOCH_3_1_MARKER: u8 = 0x0c;
/// NOTE: it has to be 0x0d because a prior release of 3.1 with 0x0c before activation had a
/// consensus bug. This forces miners with this buggy release off the network if they are still
/// running it prior to 3.1 activation.
pub static STACKS_EPOCH_3_1_MARKER: u8 = 0x0d;

#[test]
fn test_ord_for_stacks_epoch() {
Expand Down

0 comments on commit 464b796

Please sign in to comment.